2 * gstvaapisink.c - VA-API video sink
4 * Copyright (C) 2010-2011 Splitted-Desktop Systems
5 * Author: Gwenole Beauchesne <gwenole.beauchesne@splitted-desktop.com>
6 * Copyright (C) 2011-2014 Intel Corporation
7 * Author: Gwenole Beauchesne <gwenole.beauchesne@intel.com>
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public License
11 * as published by the Free Software Foundation; either version 2.1
12 * of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free
21 * Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
22 * Boston, MA 02110-1301 USA
26 * SECTION:gstvaapisink
27 * @short_description: A VA-API based videosink
29 * vaapisink renders video frames to a drawable (X #Window) on a local
30 * display using the Video Acceleration (VA) API. The element will
31 * create its own internal window and render into it.
34 #include "gst/vaapi/sysdeps.h"
36 #include <gst/video/video.h>
38 #include <gst/vaapi/gstvaapivalue.h>
40 # include <gst/vaapi/gstvaapidisplay_drm.h>
43 # include <gst/vaapi/gstvaapidisplay_x11.h>
44 # include <gst/vaapi/gstvaapiwindow_x11.h>
47 # include <gst/vaapi/gstvaapidisplay_glx.h>
48 # include <gst/vaapi/gstvaapiwindow_glx.h>
51 # include <gst/vaapi/gstvaapidisplay_wayland.h>
52 # include <gst/vaapi/gstvaapiwindow_wayland.h>
55 /* Supported interfaces */
56 #if GST_CHECK_VERSION(1,0,0)
57 # include <gst/video/videooverlay.h>
59 # include <gst/interfaces/xoverlay.h>
61 # define GST_TYPE_VIDEO_OVERLAY GST_TYPE_X_OVERLAY
62 # define GST_VIDEO_OVERLAY GST_X_OVERLAY
63 # define GstVideoOverlay GstXOverlay
64 # define GstVideoOverlayInterface GstXOverlayClass
66 # define gst_video_overlay_prepare_window_handle(sink) \
67 gst_x_overlay_prepare_xwindow_id(sink)
68 # define gst_video_overlay_got_window_handle(sink, window_handle) \
69 gst_x_overlay_got_window_handle(sink, window_handle)
72 #include "gstvaapisink.h"
73 #include "gstvaapipluginutil.h"
74 #include "gstvaapivideometa.h"
75 #if GST_CHECK_VERSION(1,0,0)
76 #include "gstvaapivideobufferpool.h"
77 #include "gstvaapivideomemory.h"
80 #define GST_PLUGIN_NAME "vaapisink"
81 #define GST_PLUGIN_DESC "A VA-API based videosink"
83 GST_DEBUG_CATEGORY_STATIC(gst_debug_vaapisink);
84 #define GST_CAT_DEFAULT gst_debug_vaapisink
86 /* Default template */
87 static const char gst_vaapisink_sink_caps_str[] =
88 #if GST_CHECK_VERSION(1,1,0)
89 GST_VIDEO_CAPS_MAKE_WITH_FEATURES(
90 GST_CAPS_FEATURE_MEMORY_VAAPI_SURFACE, "{ ENCODED, NV12, I420, YV12 }") ";"
91 GST_VIDEO_CAPS_MAKE(GST_VIDEO_FORMATS_ALL);
93 #if GST_CHECK_VERSION(1,0,0)
94 GST_VIDEO_CAPS_MAKE(GST_VIDEO_FORMATS_ALL) "; "
97 "width = (int) [ 1, MAX ], "
98 "height = (int) [ 1, MAX ]; "
100 GST_VAAPI_SURFACE_CAPS;
103 static GstStaticPadTemplate gst_vaapisink_sink_factory =
104 GST_STATIC_PAD_TEMPLATE(
108 GST_STATIC_CAPS(gst_vaapisink_sink_caps_str));
111 gst_vaapisink_has_interface(GstVaapiPluginBase *plugin, GType type)
113 return type == GST_TYPE_VIDEO_OVERLAY;
117 gst_vaapisink_video_overlay_iface_init(GstVideoOverlayInterface *iface);
119 G_DEFINE_TYPE_WITH_CODE(
123 GST_VAAPI_PLUGIN_BASE_INIT_INTERFACES
124 G_IMPLEMENT_INTERFACE(GST_TYPE_VIDEO_OVERLAY,
125 gst_vaapisink_video_overlay_iface_init))
137 PROP_FORCE_ASPECT_RATIO,
141 #define DEFAULT_DISPLAY_TYPE GST_VAAPI_DISPLAY_TYPE_ANY
142 #define DEFAULT_ROTATION GST_VAAPI_ROTATION_0
144 static inline gboolean
145 gst_vaapisink_ensure_display(GstVaapiSink *sink);
147 /* GstVideoOverlay interface */
151 gst_vaapisink_ensure_window_xid(GstVaapiSink *sink, guintptr window_id);
155 gst_vaapisink_show_frame(GstBaseSink *base_sink, GstBuffer *buffer);
158 gst_vaapisink_video_overlay_set_window_handle(GstVideoOverlay *overlay,
161 GstVaapiSink * const sink = GST_VAAPISINK(overlay);
162 GstVaapiDisplayType display_type;
164 if (!gst_vaapisink_ensure_display(sink))
166 display_type = GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink);
168 /* Disable GLX rendering when vaapisink is using a foreign X
169 window. It's pretty much useless */
170 if (display_type == GST_VAAPI_DISPLAY_TYPE_GLX) {
171 display_type = GST_VAAPI_DISPLAY_TYPE_X11;
172 gst_vaapi_plugin_base_set_display_type(GST_VAAPI_PLUGIN_BASE(sink),
176 sink->foreign_window = TRUE;
178 switch (display_type) {
180 case GST_VAAPI_DISPLAY_TYPE_X11:
181 gst_vaapisink_ensure_window_xid(sink, window);
190 gst_vaapisink_video_overlay_set_render_rectangle(
191 GstVideoOverlay *overlay,
198 GstVaapiSink * const sink = GST_VAAPISINK(overlay);
199 GstVaapiRectangle * const display_rect = &sink->display_rect;
203 display_rect->width = width;
204 display_rect->height = height;
206 GST_DEBUG("render rect (%d,%d):%ux%u",
207 display_rect->x, display_rect->y,
208 display_rect->width, display_rect->height);
212 gst_vaapisink_video_overlay_expose(GstVideoOverlay *overlay)
214 GstVaapiSink * const sink = GST_VAAPISINK(overlay);
216 if (sink->video_buffer)
217 gst_vaapisink_show_frame(GST_BASE_SINK_CAST(sink),
218 gst_buffer_ref(sink->video_buffer));
222 gst_vaapisink_video_overlay_iface_init(GstVideoOverlayInterface *iface)
224 iface->set_window_handle = gst_vaapisink_video_overlay_set_window_handle;
225 iface->set_render_rectangle = gst_vaapisink_video_overlay_set_render_rectangle;
226 iface->expose = gst_vaapisink_video_overlay_expose;
230 gst_vaapisink_destroy(GstVaapiSink *sink)
232 gst_buffer_replace(&sink->video_buffer, NULL);
234 gst_vaapi_texture_replace(&sink->texture, NULL);
236 gst_caps_replace(&sink->caps, NULL);
240 /* Checks whether a ConfigureNotify event is in the queue */
241 typedef struct _ConfigureNotifyEventPendingArgs ConfigureNotifyEventPendingArgs;
242 struct _ConfigureNotifyEventPendingArgs {
250 configure_notify_event_pending_cb(Display *dpy, XEvent *xev, XPointer arg)
252 ConfigureNotifyEventPendingArgs * const args =
253 (ConfigureNotifyEventPendingArgs *)arg;
255 if (xev->type == ConfigureNotify &&
256 xev->xconfigure.window == args->window &&
257 xev->xconfigure.width == args->width &&
258 xev->xconfigure.height == args->height)
261 /* XXX: this is a hack to traverse the whole queue because we
262 can't use XPeekIfEvent() since it could block */
267 configure_notify_event_pending(
274 GstVaapiDisplayX11 * const display =
275 GST_VAAPI_DISPLAY_X11(GST_VAAPI_PLUGIN_BASE_DISPLAY(sink));
276 ConfigureNotifyEventPendingArgs args;
279 args.window = window;
281 args.height = height;
284 /* XXX: don't use XPeekIfEvent() because it might block */
286 gst_vaapi_display_x11_get_display(display),
288 configure_notify_event_pending_cb, (XPointer)&args
295 get_display_type_name(GstVaapiDisplayType display_type)
297 gpointer const klass = g_type_class_peek(GST_VAAPI_TYPE_DISPLAY_TYPE);
298 GEnumValue * const e = g_enum_get_value(klass, display_type);
301 return e->value_name;
302 return "<unknown-type>";
305 static inline gboolean
306 gst_vaapisink_ensure_display(GstVaapiSink *sink)
308 return gst_vaapi_plugin_base_ensure_display(GST_VAAPI_PLUGIN_BASE(sink));
312 gst_vaapisink_display_changed(GstVaapiPluginBase *plugin)
314 GstVaapiSink * const sink = GST_VAAPISINK(plugin);
315 GstVaapiRenderMode render_mode;
317 GST_INFO("created %s %p", get_display_type_name(plugin->display_type),
321 gst_vaapi_display_get_render_mode(plugin->display, &render_mode) &&
322 render_mode == GST_VAAPI_RENDER_MODE_OVERLAY;
323 GST_DEBUG("use %s rendering mode",
324 sink->use_overlay ? "overlay" : "texture");
326 sink->use_rotation = gst_vaapi_display_has_property(plugin->display,
327 GST_VAAPI_DISPLAY_PROP_ROTATION);
331 gst_vaapisink_ensure_uploader(GstVaapiSink *sink)
333 if (!gst_vaapisink_ensure_display(sink))
335 if (!gst_vaapi_plugin_base_ensure_uploader(GST_VAAPI_PLUGIN_BASE(sink)))
341 gst_vaapisink_ensure_render_rect(GstVaapiSink *sink, guint width, guint height)
343 GstVaapiRectangle * const display_rect = &sink->display_rect;
344 guint num, den, display_par_n, display_par_d;
347 /* Return success if caps are not set yet */
351 if (!sink->keep_aspect) {
352 display_rect->width = width;
353 display_rect->height = height;
357 GST_DEBUG("force-aspect-ratio is false; distorting while scaling video");
358 GST_DEBUG("render rect (%d,%d):%ux%u",
359 display_rect->x, display_rect->y,
360 display_rect->width, display_rect->height);
364 GST_DEBUG("ensure render rect within %ux%u bounds", width, height);
366 gst_vaapi_display_get_pixel_aspect_ratio(
367 GST_VAAPI_PLUGIN_BASE_DISPLAY(sink),
368 &display_par_n, &display_par_d
370 GST_DEBUG("display pixel-aspect-ratio %d/%d",
371 display_par_n, display_par_d);
373 success = gst_video_calculate_display_ratio(
375 sink->video_width, sink->video_height,
376 sink->video_par_n, sink->video_par_d,
377 display_par_n, display_par_d
381 GST_DEBUG("video size %dx%d, calculated ratio %d/%d",
382 sink->video_width, sink->video_height, num, den);
384 display_rect->width = gst_util_uint64_scale_int(height, num, den);
385 if (display_rect->width <= width) {
386 GST_DEBUG("keeping window height");
387 display_rect->height = height;
390 GST_DEBUG("keeping window width");
391 display_rect->width = width;
392 display_rect->height =
393 gst_util_uint64_scale_int(width, den, num);
395 GST_DEBUG("scaling video to %ux%u", display_rect->width, display_rect->height);
397 g_assert(display_rect->width <= width);
398 g_assert(display_rect->height <= height);
400 display_rect->x = (width - display_rect->width) / 2;
401 display_rect->y = (height - display_rect->height) / 2;
403 GST_DEBUG("render rect (%d,%d):%ux%u",
404 display_rect->x, display_rect->y,
405 display_rect->width, display_rect->height);
410 gst_vaapisink_ensure_window_size(GstVaapiSink *sink, guint *pwidth, guint *pheight)
412 GstVaapiDisplay * const display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
413 GstVideoRectangle src_rect, dst_rect, out_rect;
414 guint num, den, display_width, display_height, display_par_n, display_par_d;
415 gboolean success, scale;
417 if (sink->foreign_window) {
418 *pwidth = sink->window_width;
419 *pheight = sink->window_height;
423 gst_vaapi_display_get_size(display, &display_width, &display_height);
424 if (sink->fullscreen) {
425 *pwidth = display_width;
426 *pheight = display_height;
430 gst_vaapi_display_get_pixel_aspect_ratio(
432 &display_par_n, &display_par_d
435 success = gst_video_calculate_display_ratio(
437 sink->video_width, sink->video_height,
438 sink->video_par_n, sink->video_par_d,
439 display_par_n, display_par_d
442 num = sink->video_par_n;
443 den = sink->video_par_d;
448 src_rect.w = gst_util_uint64_scale_int(sink->video_height, num, den);
449 src_rect.h = sink->video_height;
452 dst_rect.w = display_width;
453 dst_rect.h = display_height;
454 scale = (src_rect.w > dst_rect.w || src_rect.h > dst_rect.h);
455 gst_video_sink_center_rect(src_rect, dst_rect, &out_rect, scale);
456 *pwidth = out_rect.w;
457 *pheight = out_rect.h;
460 static inline gboolean
461 gst_vaapisink_ensure_window(GstVaapiSink *sink, guint width, guint height)
463 GstVaapiDisplay * const display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
466 const GstVaapiDisplayType display_type =
467 GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink);
468 switch (display_type) {
470 case GST_VAAPI_DISPLAY_TYPE_GLX:
471 sink->window = gst_vaapi_window_glx_new(display, width, height);
472 goto notify_video_overlay_interface;
475 case GST_VAAPI_DISPLAY_TYPE_X11:
476 sink->window = gst_vaapi_window_x11_new(display, width, height);
477 notify_video_overlay_interface:
480 gst_video_overlay_got_window_handle(
481 GST_VIDEO_OVERLAY(sink),
482 gst_vaapi_window_x11_get_xid(GST_VAAPI_WINDOW_X11(sink->window))
487 case GST_VAAPI_DISPLAY_TYPE_WAYLAND:
488 sink->window = gst_vaapi_window_wayland_new(display, width, height);
492 GST_ERROR("unsupported display type %d", display_type);
496 return sink->window != NULL;
501 gst_vaapisink_ensure_window_xid(GstVaapiSink *sink, guintptr window_id)
503 GstVaapiDisplay *display;
505 unsigned int width, height, border_width, depth;
509 if (!gst_vaapisink_ensure_display(sink))
511 display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
513 gst_vaapi_display_lock(display);
515 gst_vaapi_display_x11_get_display(GST_VAAPI_DISPLAY_X11(display)),
518 &x, &y, &width, &height, &border_width, &depth
520 gst_vaapi_display_unlock(display);
522 if ((width != sink->window_width || height != sink->window_height) &&
523 !configure_notify_event_pending(sink, xid, width, height)) {
524 if (!gst_vaapisink_ensure_render_rect(sink, width, height))
526 sink->window_width = width;
527 sink->window_height = height;
531 gst_vaapi_window_x11_get_xid(GST_VAAPI_WINDOW_X11(sink->window)) == xid)
534 gst_vaapi_window_replace(&sink->window, NULL);
536 switch (GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink)) {
538 case GST_VAAPI_DISPLAY_TYPE_GLX:
539 sink->window = gst_vaapi_window_glx_new_with_xid(display, xid);
542 case GST_VAAPI_DISPLAY_TYPE_X11:
543 sink->window = gst_vaapi_window_x11_new_with_xid(display, xid);
546 GST_ERROR("unsupported display type %d",
547 GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink));
550 return sink->window != NULL;
555 gst_vaapisink_ensure_rotation(GstVaapiSink *sink, gboolean recalc_display_rect)
557 GstVaapiDisplay * const display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
558 gboolean success = FALSE;
560 g_return_val_if_fail(display, FALSE);
562 if (sink->rotation == sink->rotation_req)
565 if (!sink->use_rotation) {
566 GST_WARNING("VA display does not support rotation");
570 gst_vaapi_display_lock(display);
571 success = gst_vaapi_display_set_rotation(display, sink->rotation_req);
572 gst_vaapi_display_unlock(display);
574 GST_ERROR("failed to change VA display rotation mode");
578 if (((sink->rotation + sink->rotation_req) % 180) == 90) {
579 /* Orientation changed */
580 G_PRIMITIVE_SWAP(guint, sink->video_width, sink->video_height);
581 G_PRIMITIVE_SWAP(gint, sink->video_par_n, sink->video_par_d);
584 if (recalc_display_rect && !sink->foreign_window)
585 gst_vaapisink_ensure_render_rect(sink, sink->window_width,
586 sink->window_height);
590 sink->rotation = sink->rotation_req;
595 gst_vaapisink_start(GstBaseSink *base_sink)
597 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
599 if (!gst_vaapi_plugin_base_open(GST_VAAPI_PLUGIN_BASE(sink)))
601 return gst_vaapisink_ensure_uploader(sink);
605 gst_vaapisink_stop(GstBaseSink *base_sink)
607 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
609 gst_buffer_replace(&sink->video_buffer, NULL);
610 #if GST_CHECK_VERSION(1,0,0)
611 g_clear_object(&sink->video_buffer_pool);
613 gst_vaapi_window_replace(&sink->window, NULL);
615 gst_vaapi_plugin_base_close(GST_VAAPI_PLUGIN_BASE(sink));
620 gst_vaapisink_get_caps_impl(GstBaseSink *base_sink)
622 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
623 GstCaps *out_caps, *yuv_caps;
625 #if GST_CHECK_VERSION(1,1,0)
626 out_caps = gst_static_pad_template_get_caps(&gst_vaapisink_sink_factory);
628 out_caps = gst_caps_from_string(GST_VAAPI_SURFACE_CAPS);
633 if (gst_vaapisink_ensure_uploader(sink)) {
634 yuv_caps = GST_VAAPI_PLUGIN_BASE_UPLOADER_CAPS(sink);
636 out_caps = gst_caps_make_writable(out_caps);
637 gst_caps_append(out_caps, gst_caps_copy(yuv_caps));
643 #if GST_CHECK_VERSION(1,0,0)
644 static inline GstCaps *
645 gst_vaapisink_get_caps(GstBaseSink *base_sink, GstCaps *filter)
647 GstCaps *caps, *out_caps;
649 caps = gst_vaapisink_get_caps_impl(base_sink);
650 if (caps && filter) {
651 out_caps = gst_caps_intersect_full(caps, filter,
652 GST_CAPS_INTERSECT_FIRST);
653 gst_caps_unref(caps);
660 #define gst_vaapisink_get_caps gst_vaapisink_get_caps_impl
664 update_colorimetry(GstVaapiSink *sink, GstVideoColorimetry *cinfo)
666 #if GST_CHECK_VERSION(1,0,0)
667 if (gst_video_colorimetry_matches(cinfo,
668 GST_VIDEO_COLORIMETRY_BT601))
669 sink->color_standard = GST_VAAPI_COLOR_STANDARD_ITUR_BT_601;
670 else if (gst_video_colorimetry_matches(cinfo,
671 GST_VIDEO_COLORIMETRY_BT709))
672 sink->color_standard = GST_VAAPI_COLOR_STANDARD_ITUR_BT_709;
673 else if (gst_video_colorimetry_matches(cinfo,
674 GST_VIDEO_COLORIMETRY_SMPTE240M))
675 sink->color_standard = GST_VAAPI_COLOR_STANDARD_SMPTE_240M;
677 sink->color_standard = 0;
679 GST_DEBUG("colorimetry %s", gst_video_colorimetry_to_string(cinfo));
684 gst_vaapisink_set_caps(GstBaseSink *base_sink, GstCaps *caps)
686 GstVaapiPluginBase * const plugin = GST_VAAPI_PLUGIN_BASE(base_sink);
687 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
688 GstVideoInfo * const vip = GST_VAAPI_PLUGIN_BASE_SINK_PAD_INFO(sink);
689 GstVaapiDisplay *display;
690 guint win_width, win_height;
692 if (!gst_vaapisink_ensure_display(sink))
694 display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
697 if (GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink) == GST_VAAPI_DISPLAY_TYPE_DRM)
701 if (!gst_vaapi_plugin_base_set_caps(plugin, caps, NULL))
704 sink->video_width = GST_VIDEO_INFO_WIDTH(vip);
705 sink->video_height = GST_VIDEO_INFO_HEIGHT(vip);
706 sink->video_par_n = GST_VIDEO_INFO_PAR_N(vip);
707 sink->video_par_d = GST_VIDEO_INFO_PAR_D(vip);
708 GST_DEBUG("video pixel-aspect-ratio %d/%d",
709 sink->video_par_n, sink->video_par_d);
711 update_colorimetry(sink, &vip->colorimetry);
712 gst_caps_replace(&sink->caps, caps);
714 gst_vaapisink_ensure_rotation(sink, FALSE);
716 gst_vaapisink_ensure_window_size(sink, &win_width, &win_height);
718 if (!sink->foreign_window || sink->fullscreen)
719 gst_vaapi_window_set_size(sink->window, win_width, win_height);
722 gst_vaapi_display_lock(display);
723 gst_video_overlay_prepare_window_handle(GST_VIDEO_OVERLAY(sink));
724 gst_vaapi_display_unlock(display);
727 if (!gst_vaapisink_ensure_window(sink, win_width, win_height))
729 gst_vaapi_window_set_fullscreen(sink->window, sink->fullscreen);
730 gst_vaapi_window_show(sink->window);
731 gst_vaapi_window_get_size(sink->window, &win_width, &win_height);
733 sink->window_width = win_width;
734 sink->window_height = win_height;
735 GST_DEBUG("window size %ux%u", win_width, win_height);
737 return gst_vaapisink_ensure_render_rect(sink, win_width, win_height);
742 render_background(GstVaapiSink *sink)
744 /* Original code from Mirco Muller (MacSlow):
745 <http://cgit.freedesktop.org/~macslow/gl-gst-player/> */
746 GLfloat fStartX = 0.0f;
747 GLfloat fStartY = 0.0f;
748 GLfloat fWidth = (GLfloat)sink->window_width;
749 GLfloat fHeight = (GLfloat)sink->window_height;
751 glClear(GL_COLOR_BUFFER_BIT);
754 /* top third, darker grey to white */
755 glColor3f(0.85f, 0.85f, 0.85f);
756 glVertex3f(fStartX, fStartY, 0.0f);
757 glColor3f(0.85f, 0.85f, 0.85f);
758 glVertex3f(fStartX + fWidth, fStartY, 0.0f);
759 glColor3f(1.0f, 1.0f, 1.0f);
760 glVertex3f(fStartX + fWidth, fStartY + fHeight / 3.0f, 0.0f);
761 glColor3f(1.0f, 1.0f, 1.0f);
762 glVertex3f(fStartX, fStartY + fHeight / 3.0f, 0.0f);
764 /* middle third, just plain white */
765 glColor3f(1.0f, 1.0f, 1.0f);
766 glVertex3f(fStartX, fStartY + fHeight / 3.0f, 0.0f);
767 glVertex3f(fStartX + fWidth, fStartY + fHeight / 3.0f, 0.0f);
768 glVertex3f(fStartX + fWidth, fStartY + 2.0f * fHeight / 3.0f, 0.0f);
769 glVertex3f(fStartX, fStartY + 2.0f * fHeight / 3.0f, 0.0f);
771 /* bottom third, white to lighter grey */
772 glColor3f(1.0f, 1.0f, 1.0f);
773 glVertex3f(fStartX, fStartY + 2.0f * fHeight / 3.0f, 0.0f);
774 glColor3f(1.0f, 1.0f, 1.0f);
775 glVertex3f(fStartX + fWidth, fStartY + 2.0f * fHeight / 3.0f, 0.0f);
776 glColor3f(0.62f, 0.66f, 0.69f);
777 glVertex3f(fStartX + fWidth, fStartY + fHeight, 0.0f);
778 glColor3f(0.62f, 0.66f, 0.69f);
779 glVertex3f(fStartX, fStartY + fHeight, 0.0f);
785 render_frame(GstVaapiSink *sink, GstVaapiSurface *surface,
786 const GstVaapiRectangle *surface_rect)
788 const guint x1 = sink->display_rect.x;
789 const guint x2 = sink->display_rect.x + sink->display_rect.width;
790 const guint y1 = sink->display_rect.y;
791 const guint y2 = sink->display_rect.y + sink->display_rect.height;
792 gfloat tx1, tx2, ty1, ty2;
796 gst_vaapi_surface_get_size(surface, &width, &height);
797 tx1 = (gfloat)surface_rect->x / width;
798 ty1 = (gfloat)surface_rect->y / height;
799 tx2 = (gfloat)(surface_rect->x + surface_rect->width) / width;
800 ty2 = (gfloat)(surface_rect->y + surface_rect->height) / height;
809 glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
812 glTexCoord2f(tx1, ty1); glVertex2i(x1, y1);
813 glTexCoord2f(tx1, ty2); glVertex2i(x1, y2);
814 glTexCoord2f(tx2, ty2); glVertex2i(x2, y2);
815 glTexCoord2f(tx2, ty1); glVertex2i(x2, y1);
821 render_reflection(GstVaapiSink *sink)
823 const guint x1 = sink->display_rect.x;
824 const guint x2 = sink->display_rect.x + sink->display_rect.width;
825 const guint y1 = sink->display_rect.y;
826 const guint rh = sink->display_rect.height / 5;
827 GLfloat ry = 1.0f - (GLfloat)rh / (GLfloat)sink->display_rect.height;
831 glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
832 glTexCoord2f(0.0f, 1.0f); glVertex2i(x1, y1);
833 glTexCoord2f(1.0f, 1.0f); glVertex2i(x2, y1);
835 glColor4f(1.0f, 1.0f, 1.0f, 0.0f);
836 glTexCoord2f(1.0f, ry); glVertex2i(x2, y1 + rh);
837 glTexCoord2f(0.0f, ry); glVertex2i(x1, y1 + rh);
843 gst_vaapisink_ensure_texture(GstVaapiSink *sink, GstVaapiSurface *surface)
845 GstVaapiDisplay * display = GST_VAAPI_PLUGIN_BASE_DISPLAY(sink);
846 GstVideoRectangle tex_rect, dis_rect, out_rect;
852 gst_vaapi_surface_get_size(surface, &width, &height);
858 gst_vaapi_display_get_size(display, &width, &height);
864 gst_video_sink_center_rect(tex_rect, dis_rect, &out_rect, TRUE);
866 /* XXX: use surface size for now since some VA drivers have issues
867 with downscaling to the provided texture size. i.e. we should be
868 using the resulting out_rect size, which preserves the aspect
869 ratio of the surface */
872 GST_INFO("texture size %ux%u", width, height);
874 sink->texture = gst_vaapi_texture_new(display,
875 GL_TEXTURE_2D, GL_BGRA, width, height);
876 return sink->texture != NULL;
880 gst_vaapisink_show_frame_glx(
882 GstVaapiSurface *surface,
883 const GstVaapiRectangle *surface_rect,
887 GstVaapiWindowGLX *window;
893 window = GST_VAAPI_WINDOW_GLX(sink->window);
895 gst_vaapi_window_glx_make_current(window);
896 if (!gst_vaapisink_ensure_texture(sink, surface))
897 goto error_create_texture;
898 if (!gst_vaapi_texture_put_surface(sink->texture, surface, flags))
899 goto error_transfer_surface;
901 target = gst_vaapi_texture_get_target(sink->texture);
902 texture = gst_vaapi_texture_get_id(sink->texture);
903 if (target != GL_TEXTURE_2D || !texture)
906 if (sink->use_reflection)
907 render_background(sink);
910 glBindTexture(target, texture);
912 if (sink->use_reflection) {
914 glRotatef(20.0f, 0.0f, 1.0f, 0.0f);
915 glTranslatef(50.0f, 0.0f, 0.0f);
917 render_frame(sink, surface, surface_rect);
918 if (sink->use_reflection) {
920 glTranslatef(0.0, (GLfloat)sink->display_rect.height + 5.0f, 0.0f);
921 render_reflection(sink);
926 glBindTexture(target, 0);
928 gst_vaapi_window_glx_swap_buffers(window);
932 error_create_texture:
934 GST_DEBUG("could not create VA/GLX texture");
937 error_transfer_surface:
939 GST_DEBUG("could not transfer VA surface to texture");
945 static inline gboolean
946 gst_vaapisink_put_surface(
948 GstVaapiSurface *surface,
949 const GstVaapiRectangle *surface_rect,
956 if (!gst_vaapi_window_put_surface(sink->window, surface,
957 surface_rect, &sink->display_rect, flags)) {
958 GST_DEBUG("could not render VA surface");
965 gst_vaapisink_show_frame(GstBaseSink *base_sink, GstBuffer *src_buffer)
967 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
968 GstVaapiVideoMeta *meta;
969 GstVaapiSurfaceProxy *proxy;
970 GstVaapiSurface *surface;
974 GstVaapiRectangle *surface_rect = NULL;
975 #if GST_CHECK_VERSION(1,0,0)
976 GstVaapiRectangle tmp_rect;
981 #if GST_CHECK_VERSION(1,0,0)
982 GstVideoCropMeta * const crop_meta =
983 gst_buffer_get_video_crop_meta(src_buffer);
985 surface_rect = &tmp_rect;
986 surface_rect->x = crop_meta->x;
987 surface_rect->y = crop_meta->y;
988 surface_rect->width = crop_meta->width;
989 surface_rect->height = crop_meta->height;
993 ret = gst_vaapi_plugin_base_get_input_buffer(GST_VAAPI_PLUGIN_BASE(sink),
994 src_buffer, &buffer);
995 if (ret != GST_FLOW_OK && ret != GST_FLOW_NOT_SUPPORTED)
998 meta = gst_buffer_get_vaapi_video_meta(buffer);
999 GST_VAAPI_PLUGIN_BASE_DISPLAY_REPLACE(sink,
1000 gst_vaapi_video_meta_get_display(meta));
1002 gst_vaapisink_ensure_rotation(sink, TRUE);
1004 proxy = gst_vaapi_video_meta_get_surface_proxy(meta);
1008 /* Valide view component to display */
1009 view_id = GST_VAAPI_SURFACE_PROXY_VIEW_ID(proxy);
1010 if (G_UNLIKELY(sink->view_id == -1))
1011 sink->view_id = view_id;
1012 else if (sink->view_id != view_id) {
1013 gst_buffer_unref(buffer);
1017 surface = gst_vaapi_video_meta_get_surface(meta);
1021 GST_DEBUG("render surface %" GST_VAAPI_ID_FORMAT,
1022 GST_VAAPI_ID_ARGS(gst_vaapi_surface_get_id(surface)));
1025 surface_rect = (GstVaapiRectangle *)
1026 gst_vaapi_video_meta_get_render_rect(meta);
1029 GST_DEBUG("render rect (%d,%d), size %ux%u",
1030 surface_rect->x, surface_rect->y,
1031 surface_rect->width, surface_rect->height);
1033 flags = gst_vaapi_video_meta_get_render_flags(meta);
1035 /* Append default color standard obtained from caps if none was
1036 available on a per-buffer basis */
1037 if (!(flags & GST_VAAPI_COLOR_STANDARD_MASK))
1038 flags |= sink->color_standard;
1040 if (!gst_vaapi_apply_composition(surface, src_buffer))
1041 GST_WARNING("could not update subtitles");
1043 switch (GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink)) {
1045 case GST_VAAPI_DISPLAY_TYPE_DRM:
1050 case GST_VAAPI_DISPLAY_TYPE_GLX:
1052 goto put_surface_x11;
1053 success = gst_vaapisink_show_frame_glx(sink, surface, surface_rect,
1058 case GST_VAAPI_DISPLAY_TYPE_X11:
1060 success = gst_vaapisink_put_surface(sink, surface, surface_rect, flags);
1064 case GST_VAAPI_DISPLAY_TYPE_WAYLAND:
1065 success = gst_vaapisink_put_surface(sink, surface, surface_rect, flags);
1069 GST_ERROR("unsupported display type %d",
1070 GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink));
1077 /* Retain VA surface until the next one is displayed */
1078 gst_buffer_replace(&sink->video_buffer, buffer);
1079 gst_buffer_unref(buffer);
1083 gst_buffer_unref(buffer);
1084 return GST_FLOW_EOS;
1087 #if GST_CHECK_VERSION(1,0,0)
1089 gst_vaapisink_propose_allocation(GstBaseSink *base_sink, GstQuery *query)
1091 GstVaapiPluginBase * const plugin = GST_VAAPI_PLUGIN_BASE(base_sink);
1093 if (!gst_vaapi_plugin_base_propose_allocation(plugin, query))
1096 gst_query_add_allocation_meta(query, GST_VIDEO_CROP_META_API_TYPE, NULL);
1097 gst_query_add_allocation_meta(query,
1098 GST_VIDEO_OVERLAY_COMPOSITION_META_API_TYPE, NULL);
1102 static GstFlowReturn
1103 gst_vaapisink_buffer_alloc(GstBaseSink *base_sink, guint64 offset, guint size,
1104 GstCaps *caps, GstBuffer **outbuf_ptr)
1106 return gst_vaapi_plugin_base_allocate_input_buffer(
1107 GST_VAAPI_PLUGIN_BASE(base_sink), caps, outbuf_ptr);
1112 gst_vaapisink_query(GstBaseSink *base_sink, GstQuery *query)
1114 GstVaapiSink * const sink = GST_VAAPISINK(base_sink);
1116 GST_INFO_OBJECT(sink, "query type %s", GST_QUERY_TYPE_NAME(query));
1118 if (gst_vaapi_reply_to_query(query, GST_VAAPI_PLUGIN_BASE_DISPLAY(sink))) {
1119 GST_DEBUG("sharing display %p", GST_VAAPI_PLUGIN_BASE_DISPLAY(sink));
1123 return GST_BASE_SINK_CLASS(gst_vaapisink_parent_class)->query(base_sink,
1128 gst_vaapisink_finalize(GObject *object)
1130 gst_vaapisink_destroy(GST_VAAPISINK(object));
1132 gst_vaapi_plugin_base_finalize(GST_VAAPI_PLUGIN_BASE(object));
1133 G_OBJECT_CLASS(gst_vaapisink_parent_class)->finalize(object);
1137 gst_vaapisink_set_property(
1140 const GValue *value,
1144 GstVaapiSink * const sink = GST_VAAPISINK(object);
1147 case PROP_DISPLAY_TYPE:
1148 gst_vaapi_plugin_base_set_display_type(GST_VAAPI_PLUGIN_BASE(sink),
1149 g_value_get_enum(value));
1151 case PROP_DISPLAY_NAME:
1152 gst_vaapi_plugin_base_set_display_name(GST_VAAPI_PLUGIN_BASE(sink),
1153 g_value_get_string(value));
1155 case PROP_FULLSCREEN:
1156 sink->fullscreen = g_value_get_boolean(value);
1158 case PROP_SYNCHRONOUS:
1159 sink->synchronous = g_value_get_boolean(value);
1162 sink->view_id = g_value_get_int(value);
1165 sink->use_glx = g_value_get_boolean(value);
1167 case PROP_USE_REFLECTION:
1168 sink->use_reflection = g_value_get_boolean(value);
1171 sink->rotation_req = g_value_get_enum(value);
1173 case PROP_FORCE_ASPECT_RATIO:
1174 sink->keep_aspect = g_value_get_boolean(value);
1177 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
1183 gst_vaapisink_get_property(
1190 GstVaapiSink * const sink = GST_VAAPISINK(object);
1193 case PROP_DISPLAY_TYPE:
1194 g_value_set_enum(value, GST_VAAPI_PLUGIN_BASE_DISPLAY_TYPE(sink));
1196 case PROP_DISPLAY_NAME:
1197 g_value_set_string(value, GST_VAAPI_PLUGIN_BASE_DISPLAY_NAME(sink));
1199 case PROP_FULLSCREEN:
1200 g_value_set_boolean(value, sink->fullscreen);
1202 case PROP_SYNCHRONOUS:
1203 g_value_set_boolean(value, sink->synchronous);
1206 g_value_set_int(value, sink->view_id);
1209 g_value_set_boolean(value, sink->use_glx);
1211 case PROP_USE_REFLECTION:
1212 g_value_set_boolean(value, sink->use_reflection);
1215 g_value_set_enum(value, sink->rotation);
1217 case PROP_FORCE_ASPECT_RATIO:
1218 g_value_set_boolean(value, sink->keep_aspect);
1221 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
1227 gst_vaapisink_set_bus(GstElement *element, GstBus *bus)
1229 /* Make sure to allocate a VA display in the sink element first,
1230 so that upstream elements could query a display that was
1231 allocated here, and that exactly matches what the user
1232 requested through the "display" property */
1233 if (!GST_ELEMENT_BUS(element) && bus)
1234 gst_vaapisink_ensure_display(GST_VAAPISINK(element));
1236 GST_ELEMENT_CLASS(gst_vaapisink_parent_class)->set_bus(element, bus);
1240 gst_vaapisink_class_init(GstVaapiSinkClass *klass)
1242 GObjectClass * const object_class = G_OBJECT_CLASS(klass);
1243 GstElementClass * const element_class = GST_ELEMENT_CLASS(klass);
1244 GstBaseSinkClass * const basesink_class = GST_BASE_SINK_CLASS(klass);
1245 GstVaapiPluginBaseClass * const base_plugin_class =
1246 GST_VAAPI_PLUGIN_BASE_CLASS(klass);
1247 GstPadTemplate *pad_template;
1249 GST_DEBUG_CATEGORY_INIT(gst_debug_vaapisink,
1250 GST_PLUGIN_NAME, 0, GST_PLUGIN_DESC);
1252 gst_vaapi_plugin_base_class_init(base_plugin_class);
1253 base_plugin_class->has_interface = gst_vaapisink_has_interface;
1254 base_plugin_class->display_changed = gst_vaapisink_display_changed;
1256 object_class->finalize = gst_vaapisink_finalize;
1257 object_class->set_property = gst_vaapisink_set_property;
1258 object_class->get_property = gst_vaapisink_get_property;
1260 basesink_class->start = gst_vaapisink_start;
1261 basesink_class->stop = gst_vaapisink_stop;
1262 basesink_class->get_caps = gst_vaapisink_get_caps;
1263 basesink_class->set_caps = gst_vaapisink_set_caps;
1264 basesink_class->preroll = gst_vaapisink_show_frame;
1265 basesink_class->render = gst_vaapisink_show_frame;
1266 basesink_class->query = gst_vaapisink_query;
1267 #if GST_CHECK_VERSION(1,0,0)
1268 basesink_class->propose_allocation = gst_vaapisink_propose_allocation;
1270 basesink_class->buffer_alloc = gst_vaapisink_buffer_alloc;
1273 element_class->set_bus = gst_vaapisink_set_bus;
1274 gst_element_class_set_static_metadata(element_class,
1278 "Gwenole Beauchesne <gwenole.beauchesne@intel.com>");
1280 pad_template = gst_static_pad_template_get(&gst_vaapisink_sink_factory);
1281 gst_element_class_add_pad_template(element_class, pad_template);
1283 g_object_class_install_property
1286 g_param_spec_enum("display",
1288 "display type to use",
1289 GST_VAAPI_TYPE_DISPLAY_TYPE,
1290 GST_VAAPI_DISPLAY_TYPE_ANY,
1291 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1293 g_object_class_install_property
1296 g_param_spec_string("display-name",
1298 "display name to use",
1300 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1303 g_object_class_install_property
1306 g_param_spec_boolean("use-glx",
1308 "Enables OpenGL rendering",
1310 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1312 g_object_class_install_property
1314 PROP_USE_REFLECTION,
1315 g_param_spec_boolean("use-reflection",
1316 "Reflection effect",
1317 "Enables OpenGL reflection effect",
1319 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1322 g_object_class_install_property
1325 g_param_spec_boolean("fullscreen",
1327 "Requests window in fullscreen state",
1329 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1332 * GstVaapiSink:synchronous:
1334 * When enabled, runs the X display in synchronous mode. Note that
1335 * this is used only for debugging.
1337 g_object_class_install_property
1340 g_param_spec_boolean("synchronous",
1342 "Toggles X display synchronous mode",
1344 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1347 * GstVaapiSink:rotation:
1349 * The VA display rotation mode, expressed as a #GstVaapiRotation.
1351 g_object_class_install_property
1354 g_param_spec_enum(GST_VAAPI_DISPLAY_PROP_ROTATION,
1356 "The display rotation mode",
1357 GST_VAAPI_TYPE_ROTATION,
1359 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1362 * GstVaapiSink:force-aspect-ratio:
1364 * When enabled, scaling respects video aspect ratio; when disabled, the
1365 * video is distorted to fit the window.
1367 g_object_class_install_property
1369 PROP_FORCE_ASPECT_RATIO,
1370 g_param_spec_boolean("force-aspect-ratio",
1371 "Force aspect ratio",
1372 "When enabled, scaling will respect original aspect ratio",
1374 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
1377 * GstVaapiSink:view-id:
1379 * When not set to -1, the displayed frame will always be the one
1380 * that matches the view-id of the very first displayed frame. Any
1381 * other number will indicate the desire to display the supplied
1384 g_object_class_install_property
1387 g_param_spec_int("view-id",
1389 "ID of the view component of interest to display",
1391 G_PARAM_READWRITE));
1395 gst_vaapisink_init(GstVaapiSink *sink)
1397 GstVaapiPluginBase * const plugin = GST_VAAPI_PLUGIN_BASE(sink);
1399 gst_vaapi_plugin_base_init(plugin, GST_CAT_DEFAULT);
1400 gst_vaapi_plugin_base_set_display_type(plugin, DEFAULT_DISPLAY_TYPE);
1403 sink->window = NULL;
1404 sink->window_width = 0;
1405 sink->window_height = 0;
1406 sink->texture = NULL;
1407 sink->video_buffer = NULL;
1408 sink->video_width = 0;
1409 sink->video_height = 0;
1410 sink->video_par_n = 1;
1411 sink->video_par_d = 1;
1413 sink->foreign_window = FALSE;
1414 sink->fullscreen = FALSE;
1415 sink->synchronous = FALSE;
1416 sink->rotation = DEFAULT_ROTATION;
1417 sink->rotation_req = DEFAULT_ROTATION;
1418 sink->use_reflection = FALSE;
1419 sink->use_overlay = FALSE;
1420 sink->use_rotation = FALSE;
1421 sink->keep_aspect = TRUE;
1422 gst_video_info_init(&sink->video_info);