mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 10:25:33 +00:00
wrappercamerabinsrc: Rework cropping for zoom and dimension reduction
wrappercamerabinsrc has a videocrop element to be used for zooming and for cropping when input caps is different when used with the GstPhotography interface. The zooming part needs the following elements: capsfilter ! videocrop ! videoscale ! capsfilter The capsfilters should always have the same caps to ensure the zooming is done and preserves dimensions, unless when it is needed to do more cropping due to input dimensions those caps need to be modified accordingly to preserve the output dimensions. This, however, makes it hard to get caps negotiation to work properly as we need to have different caps in the capsfilters to account for the extra cropping needed. It could be simple for fixed caps but it gets tricky with unfixed ones. To solve this, this patch splits the zooming and dimension reduction cropping into 2 separate videocrop elements. The first one does the dimension cropping, which is only needed when the GstPhotography API is used and the source provides a caps that is different than what is requested, while the second is dedicated to zoom crop only. The first part of the pipeline goes from: src ! videoconvert ! capsfilter ! videocrop ! videoscale ! capsfilter to src ! videocrop ! videoconvert ! capsfilter ! videocrop ! videoscale ! capsfilter It might add an extra overhead in the image capture as the image might need to be cropped twice but this can be solved by enabling videocrop to use crop metas so only the later one does the real cropping. It also makes the code a bit simpler.
This commit is contained in:
parent
524e536a91
commit
c6f4e4cfd8
2 changed files with 30 additions and 43 deletions
|
@ -175,6 +175,15 @@ gst_wrapper_camera_bin_src_get_property (GObject * object,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_wrapper_camera_bin_src_reset_src_zoom (GstWrapperCameraBinSrc * self)
|
||||||
|
{
|
||||||
|
if (self->src_crop) {
|
||||||
|
g_object_set (self->src_crop, "top", 0, "left", 0, "bottom", 0, "right", 0,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_wrapper_camera_bin_reset_video_src_caps (GstWrapperCameraBinSrc * self,
|
gst_wrapper_camera_bin_reset_video_src_caps (GstWrapperCameraBinSrc * self,
|
||||||
GstCaps * new_filter_caps)
|
GstCaps * new_filter_caps)
|
||||||
|
@ -272,6 +281,7 @@ gst_wrapper_camera_bin_src_imgsrc_probe (GstPad * pad, GstPadProbeInfo * info,
|
||||||
GstCaps *anycaps = gst_caps_new_any ();
|
GstCaps *anycaps = gst_caps_new_any ();
|
||||||
|
|
||||||
/* Get back to viewfinder */
|
/* Get back to viewfinder */
|
||||||
|
gst_wrapper_camera_bin_src_reset_src_zoom (self);
|
||||||
gst_wrapper_camera_bin_reset_video_src_caps (self, anycaps);
|
gst_wrapper_camera_bin_reset_video_src_caps (self, anycaps);
|
||||||
gst_wrapper_camera_bin_src_set_output (self, self->imgsrc, self->vfsrc);
|
gst_wrapper_camera_bin_src_set_output (self, self->imgsrc, self->vfsrc);
|
||||||
gst_base_camera_src_finish_capture (camerasrc);
|
gst_base_camera_src_finish_capture (camerasrc);
|
||||||
|
@ -548,6 +558,11 @@ gst_wrapper_camera_bin_src_construct_pipeline (GstBaseCameraSrc * bcamsrc)
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (self, "constructing pipeline");
|
GST_DEBUG_OBJECT (self, "constructing pipeline");
|
||||||
|
|
||||||
|
if (!(self->src_crop =
|
||||||
|
gst_camerabin_create_and_add_element (cbin, "videocrop",
|
||||||
|
"src-crop")))
|
||||||
|
goto done;
|
||||||
|
|
||||||
if (!gst_camerabin_create_and_add_element (cbin, "videoconvert",
|
if (!gst_camerabin_create_and_add_element (cbin, "videoconvert",
|
||||||
"src-videoconvert"))
|
"src-videoconvert"))
|
||||||
goto done;
|
goto done;
|
||||||
|
@ -681,11 +696,9 @@ copy_missing_fields (GQuark field_id, const GValue * value, gpointer user_data)
|
||||||
static void
|
static void
|
||||||
adapt_image_capture (GstWrapperCameraBinSrc * self, GstCaps * in_caps)
|
adapt_image_capture (GstWrapperCameraBinSrc * self, GstCaps * in_caps)
|
||||||
{
|
{
|
||||||
GstBaseCameraSrc *bcamsrc = GST_BASE_CAMERA_SRC (self);
|
|
||||||
GstStructure *in_st, *new_st, *req_st;
|
GstStructure *in_st, *new_st, *req_st;
|
||||||
gint in_width = 0, in_height = 0, req_width = 0, req_height = 0, crop = 0;
|
gint in_width = 0, in_height = 0, req_width = 0, req_height = 0, crop = 0;
|
||||||
gdouble ratio_w, ratio_h;
|
gdouble ratio_w, ratio_h;
|
||||||
GstCaps *filter_caps = NULL;
|
|
||||||
|
|
||||||
GST_LOG_OBJECT (self, "in caps: %" GST_PTR_FORMAT, in_caps);
|
GST_LOG_OBJECT (self, "in caps: %" GST_PTR_FORMAT, in_caps);
|
||||||
GST_LOG_OBJECT (self, "requested caps: %" GST_PTR_FORMAT,
|
GST_LOG_OBJECT (self, "requested caps: %" GST_PTR_FORMAT,
|
||||||
|
@ -712,53 +725,33 @@ adapt_image_capture (GstWrapperCameraBinSrc * self, GstCaps * in_caps)
|
||||||
GST_LOG_OBJECT (self, "new image capture caps: %" GST_PTR_FORMAT, new_st);
|
GST_LOG_OBJECT (self, "new image capture caps: %" GST_PTR_FORMAT, new_st);
|
||||||
|
|
||||||
/* Crop if requested aspect ratio differs from incoming frame aspect ratio */
|
/* Crop if requested aspect ratio differs from incoming frame aspect ratio */
|
||||||
if (self->src_zoom_crop) {
|
if (self->src_crop) {
|
||||||
|
gint base_crop_top = 0, base_crop_bottom = 0;
|
||||||
|
gint base_crop_left = 0, base_crop_right = 0;
|
||||||
|
|
||||||
ratio_w = (gdouble) in_width / req_width;
|
ratio_w = (gdouble) in_width / req_width;
|
||||||
ratio_h = (gdouble) in_height / req_height;
|
ratio_h = (gdouble) in_height / req_height;
|
||||||
|
|
||||||
if (ratio_w < ratio_h) {
|
if (ratio_w < ratio_h) {
|
||||||
crop = in_height - (req_height * ratio_w);
|
crop = in_height - (req_height * ratio_w);
|
||||||
self->base_crop_top = crop / 2;
|
base_crop_top = crop / 2;
|
||||||
self->base_crop_bottom = crop / 2;
|
base_crop_bottom = crop / 2;
|
||||||
} else {
|
} else {
|
||||||
crop = in_width - (req_width * ratio_h);
|
crop = in_width - (req_width * ratio_h);
|
||||||
self->base_crop_left = crop / 2;
|
base_crop_left = crop / 2;
|
||||||
self->base_crop_right += crop / 2;
|
base_crop_right += crop / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_INFO_OBJECT (self,
|
GST_INFO_OBJECT (self,
|
||||||
"setting base crop: left:%d, right:%d, top:%d, bottom:%d",
|
"setting base crop: left:%d, right:%d, top:%d, bottom:%d",
|
||||||
self->base_crop_left, self->base_crop_right, self->base_crop_top,
|
base_crop_left, base_crop_right, base_crop_top, base_crop_bottom);
|
||||||
self->base_crop_bottom);
|
g_object_set (G_OBJECT (self->src_crop),
|
||||||
g_object_set (G_OBJECT (self->src_zoom_crop),
|
"top", base_crop_top, "bottom", base_crop_bottom,
|
||||||
"top", self->base_crop_top,
|
"left", base_crop_left, "right", base_crop_right, NULL);
|
||||||
"bottom", self->base_crop_bottom,
|
|
||||||
"left", self->base_crop_left, "right", self->base_crop_right, NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update capsfilters */
|
/* Update capsfilters */
|
||||||
if (self->image_capture_caps) {
|
|
||||||
gst_caps_unref (self->image_capture_caps);
|
|
||||||
}
|
|
||||||
self->image_capture_caps = gst_caps_new_full (new_st, NULL);
|
|
||||||
set_capsfilter_caps (self, self->image_capture_caps);
|
set_capsfilter_caps (self, self->image_capture_caps);
|
||||||
|
|
||||||
/* Adjust the capsfilter before crop and videoscale elements if necessary */
|
|
||||||
if (in_width == bcamsrc->width && in_height == bcamsrc->height) {
|
|
||||||
GST_DEBUG_OBJECT (self, "no adaptation with resolution needed");
|
|
||||||
} else {
|
|
||||||
GST_DEBUG_OBJECT (self,
|
|
||||||
"changing %" GST_PTR_FORMAT " from %dx%d to %dx%d", self->src_filter,
|
|
||||||
bcamsrc->width, bcamsrc->height, in_width, in_height);
|
|
||||||
/* Apply the width and height to filter caps */
|
|
||||||
g_object_get (G_OBJECT (self->src_filter), "caps", &filter_caps, NULL);
|
|
||||||
filter_caps = gst_caps_make_writable (filter_caps);
|
|
||||||
gst_caps_set_simple (filter_caps, "width", G_TYPE_INT, in_width, "height",
|
|
||||||
G_TYPE_INT, in_height, NULL);
|
|
||||||
g_object_set (G_OBJECT (self->src_filter), "caps", filter_caps, NULL);
|
|
||||||
gst_caps_unref (filter_caps);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -932,10 +925,7 @@ set_element_zoom (GstWrapperCameraBinSrc * self, gfloat zoom)
|
||||||
GstBaseCameraSrc *bcamsrc = GST_BASE_CAMERA_SRC (self);
|
GstBaseCameraSrc *bcamsrc = GST_BASE_CAMERA_SRC (self);
|
||||||
gint w2_crop = 0, h2_crop = 0;
|
gint w2_crop = 0, h2_crop = 0;
|
||||||
GstPad *pad_zoom_sink = NULL;
|
GstPad *pad_zoom_sink = NULL;
|
||||||
gint left = self->base_crop_left;
|
gint left = 0, right = 0, top = 0, bottom = 0;
|
||||||
gint right = self->base_crop_right;
|
|
||||||
gint top = self->base_crop_top;
|
|
||||||
gint bottom = self->base_crop_bottom;
|
|
||||||
|
|
||||||
if (self->src_zoom_crop) {
|
if (self->src_zoom_crop) {
|
||||||
/* Update capsfilters to apply the zoom */
|
/* Update capsfilters to apply the zoom */
|
||||||
|
|
|
@ -103,12 +103,9 @@ struct _GstWrapperCameraBinSrc
|
||||||
/* Caps that videosrc supports */
|
/* Caps that videosrc supports */
|
||||||
GstCaps *allowed_caps;
|
GstCaps *allowed_caps;
|
||||||
|
|
||||||
/* Optional base crop for frames. Used to crop frames e.g.
|
/* Optional crop for frames. Used to crop frames e.g.
|
||||||
due to wrong aspect ratio, before the crop related to zooming. */
|
due to wrong aspect ratio. Done before the crop related to zooming. */
|
||||||
gint base_crop_top;
|
GstElement *src_crop;
|
||||||
gint base_crop_bottom;
|
|
||||||
gint base_crop_left;
|
|
||||||
gint base_crop_right;
|
|
||||||
|
|
||||||
/* Caps applied to capsfilters when in view finder mode */
|
/* Caps applied to capsfilters when in view finder mode */
|
||||||
GstCaps *view_finder_caps;
|
GstCaps *view_finder_caps;
|
||||||
|
|
Loading…
Reference in a new issue