ximagesrc: change video resolution when X11 screen gets resized

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1562>
This commit is contained in:
Jakub Adam 2021-12-17 14:57:57 +01:00 committed by GStreamer Marge Bot
parent 8c85a3143f
commit c93afcc99c
2 changed files with 45 additions and 30 deletions

View file

@ -364,12 +364,23 @@ gst_ximage_src_unlock (GstBaseSrc * basesrc)
static gboolean static gboolean
gst_ximage_src_recalc (GstXImageSrc * src) gst_ximage_src_recalc (GstXImageSrc * src)
{ {
if (!src->xcontext) XWindowAttributes attrs;
return FALSE; gboolean has_changed = FALSE;
/* Maybe later we can check the display hasn't changed size */ g_assert (src->xwindow != 0 && src->xcontext);
/* We could use XQueryPointer to get only the current window. */
return TRUE; if (XGetWindowAttributes (src->xcontext->disp, src->xwindow, &attrs)) {
if (src->xwin_width != attrs.width) {
src->xwin_width = attrs.width;
has_changed = TRUE;
}
if (src->xwin_height != attrs.height) {
src->xwin_height = attrs.height;
has_changed = TRUE;
}
}
return has_changed;
} }
#ifdef HAVE_XFIXES #ifdef HAVE_XFIXES
@ -500,7 +511,7 @@ gst_ximage_src_ximage_get (GstXImageSrc * ximagesrc)
ximagesrc->buffer_pool = g_slist_delete_link (ximagesrc->buffer_pool, ximagesrc->buffer_pool = g_slist_delete_link (ximagesrc->buffer_pool,
ximagesrc->buffer_pool); ximagesrc->buffer_pool);
if ((meta->width == ximagesrc->width) || if ((meta->width == ximagesrc->width) &&
(meta->height == ximagesrc->height)) (meta->height == ximagesrc->height))
break; break;
@ -832,12 +843,6 @@ gst_ximage_src_create (GstPushSrc * bs, GstBuffer ** buf)
GstClockTime dur; GstClockTime dur;
gint64 next_frame_no; gint64 next_frame_no;
if (!gst_ximage_src_recalc (s)) {
GST_ELEMENT_ERROR (s, RESOURCE, FAILED,
(_("Changing resolution at runtime is not yet supported.")), (NULL));
return GST_FLOW_ERROR;
}
if (s->fps_n <= 0 || s->fps_d <= 0) if (s->fps_n <= 0 || s->fps_d <= 0)
return GST_FLOW_NOT_NEGOTIATED; /* FPS must be > 0 */ return GST_FLOW_NOT_NEGOTIATED; /* FPS must be > 0 */
@ -906,7 +911,12 @@ gst_ximage_src_create (GstPushSrc * bs, GstBuffer ** buf)
s->last_frame_no = next_frame_no; s->last_frame_no = next_frame_no;
GST_OBJECT_UNLOCK (s); GST_OBJECT_UNLOCK (s);
if (gst_ximage_src_recalc (s) && !gst_base_src_negotiate (GST_BASE_SRC (s))) {
return GST_FLOW_NOT_NEGOTIATED;
}
image = gst_ximage_src_ximage_get (s); image = gst_ximage_src_ximage_get (s);
if (!image) if (!image)
return GST_FLOW_ERROR; return GST_FLOW_ERROR;
@ -944,9 +954,13 @@ gst_ximage_src_set_property (GObject * object, guint prop_id,
break; break;
case PROP_ENDX: case PROP_ENDX:
src->endx = g_value_get_uint (value); src->endx = g_value_get_uint (value);
/* property comments say 0 means right/bottom, means we can't capture
* the top left pixel alone */
src->endx_fit_to_screen = (src->endx == 0);
break; break;
case PROP_ENDY: case PROP_ENDY:
src->endy = g_value_get_uint (value); src->endy = g_value_get_uint (value);
src->endy_fit_to_screen = (src->endy == 0);
break; break;
case PROP_REMOTE: case PROP_REMOTE:
src->remote = g_value_get_boolean (value); src->remote = g_value_get_boolean (value);
@ -1069,32 +1083,21 @@ gst_ximage_src_get_caps (GstBaseSrc * bs, GstCaps * filter)
if ((!s->xcontext) && (!gst_ximage_src_open_display (s, s->display_name))) if ((!s->xcontext) && (!gst_ximage_src_open_display (s, s->display_name)))
return gst_pad_get_pad_template_caps (GST_BASE_SRC (s)->srcpad); return gst_pad_get_pad_template_caps (GST_BASE_SRC (s)->srcpad);
if (!gst_ximage_src_recalc (s)) gst_ximage_src_recalc (s);
return gst_pad_get_pad_template_caps (GST_BASE_SRC (s)->srcpad);
xcontext = s->xcontext; xcontext = s->xcontext;
width = s->xcontext->width; width = s->xwin_width;
height = s->xcontext->height; height = s->xwin_height;
if (s->xwindow != 0) {
XWindowAttributes attrs;
int status = XGetWindowAttributes (s->xcontext->disp, s->xwindow, &attrs);
if (status) {
width = attrs.width;
height = attrs.height;
}
}
/* property comments say 0 means right/bottom, means we can't capture if (s->endx_fit_to_screen)
the top left pixel alone */
if (s->endx == 0)
s->endx = width - 1; s->endx = width - 1;
if (s->endy == 0) if (s->endy_fit_to_screen)
s->endy = height - 1; s->endy = height - 1;
if (s->endx >= s->startx && s->endy >= s->starty) { if (s->endx >= s->startx && s->endy >= s->starty) {
/* this means user has put in values */ /* this means user has put in values */
if (s->startx < xcontext->width && s->endx < xcontext->width && if (s->startx < width && s->endx < width &&
s->starty < xcontext->height && s->endy < xcontext->height) { s->starty < height && s->endy < height) {
/* values are fine */ /* values are fine */
s->width = width = s->endx - s->startx + 1; s->width = width = s->endx - s->startx + 1;
s->height = height = s->endy - s->starty + 1; s->height = height = s->endy - s->starty + 1;
@ -1105,6 +1108,8 @@ gst_ximage_src_get_caps (GstBaseSrc * bs, GstCaps * filter)
s->starty = 0; s->starty = 0;
s->endx = width - 1; s->endx = width - 1;
s->endy = height - 1; s->endy = height - 1;
s->endx_fit_to_screen = FALSE;
s->endy_fit_to_screen = FALSE;
} }
} else { } else {
GST_WARNING ("User put in bogus co-ordinates, setting to full screen"); GST_WARNING ("User put in bogus co-ordinates, setting to full screen");
@ -1112,6 +1117,8 @@ gst_ximage_src_get_caps (GstBaseSrc * bs, GstCaps * filter)
s->starty = 0; s->starty = 0;
s->endx = width - 1; s->endx = width - 1;
s->endy = height - 1; s->endy = height - 1;
s->endx_fit_to_screen = FALSE;
s->endy_fit_to_screen = FALSE;
} }
GST_DEBUG ("width = %d, height=%d", width, height); GST_DEBUG ("width = %d, height=%d", width, height);
@ -1311,6 +1318,8 @@ gst_ximage_src_init (GstXImageSrc * ximagesrc)
ximagesrc->starty = 0; ximagesrc->starty = 0;
ximagesrc->endx = 0; ximagesrc->endx = 0;
ximagesrc->endy = 0; ximagesrc->endy = 0;
ximagesrc->endx_fit_to_screen = TRUE;
ximagesrc->endy_fit_to_screen = TRUE;
ximagesrc->remote = FALSE; ximagesrc->remote = FALSE;
} }

View file

@ -56,6 +56,8 @@ struct _GstXImageSrc
Window xwindow; Window xwindow;
gchar *display_name; gchar *display_name;
gint xwin_width;
gint xwin_height;
/* Window selection */ /* Window selection */
guint64 xid; guint64 xid;
@ -88,6 +90,10 @@ struct _GstXImageSrc
guint endx; guint endx;
guint endy; guint endy;
/* whether the user has put in specific coordinates for endx and/or endy */
gboolean endx_fit_to_screen;
gboolean endy_fit_to_screen;
/* whether to use remote friendly calls */ /* whether to use remote friendly calls */
gboolean remote; gboolean remote;