mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-06-06 07:28:53 +00:00
camerabin2: add videorecordingbin::video-encoder property
This commit is contained in:
parent
78dc07987e
commit
c76aeb77e5
3 changed files with 138 additions and 14 deletions
|
@ -35,6 +35,7 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "gstvideorecordingbin.h"
|
#include "gstvideorecordingbin.h"
|
||||||
|
#include "camerabingeneral.h"
|
||||||
|
|
||||||
/* prototypes */
|
/* prototypes */
|
||||||
|
|
||||||
|
@ -42,10 +43,15 @@
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
PROP_0,
|
PROP_0,
|
||||||
PROP_LOCATION
|
PROP_LOCATION,
|
||||||
|
PROP_VIDEO_ENCODER
|
||||||
};
|
};
|
||||||
|
|
||||||
#define DEFAULT_LOCATION "vidcap"
|
#define DEFAULT_LOCATION "vidcap"
|
||||||
|
#define DEFAULT_COLORSPACE "ffmpegcolorspace"
|
||||||
|
#define DEFAULT_VIDEO_ENCODER "theoraenc"
|
||||||
|
#define DEFAULT_MUXER "oggmux"
|
||||||
|
#define DEFAULT_SINK "filesink"
|
||||||
|
|
||||||
/* pad templates */
|
/* pad templates */
|
||||||
|
|
||||||
|
@ -60,11 +66,31 @@ static GstStaticPadTemplate sink_template = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||||
GST_BOILERPLATE (GstVideoRecordingBin, gst_video_recording_bin, GstBin,
|
GST_BOILERPLATE (GstVideoRecordingBin, gst_video_recording_bin, GstBin,
|
||||||
GST_TYPE_BIN);
|
GST_TYPE_BIN);
|
||||||
|
|
||||||
|
/* GObject callbacks */
|
||||||
|
static void gst_video_recording_bin_dispose (GObject * object);
|
||||||
|
static void gst_video_recording_bin_finalize (GObject * object);
|
||||||
|
|
||||||
/* Element class functions */
|
/* Element class functions */
|
||||||
static GstStateChangeReturn
|
static GstStateChangeReturn
|
||||||
gst_video_recording_bin_change_state (GstElement * element,
|
gst_video_recording_bin_change_state (GstElement * element,
|
||||||
GstStateChange trans);
|
GstStateChange trans);
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_video_recording_bin_set_video_encoder (GstVideoRecordingBin * videobin,
|
||||||
|
GstElement * encoder)
|
||||||
|
{
|
||||||
|
GST_DEBUG_OBJECT (GST_OBJECT (videobin),
|
||||||
|
"Setting video encoder %" GST_PTR_FORMAT, encoder);
|
||||||
|
|
||||||
|
if (videobin->user_video_encoder)
|
||||||
|
g_object_unref (videobin->user_video_encoder);
|
||||||
|
|
||||||
|
if (encoder)
|
||||||
|
g_object_ref (encoder);
|
||||||
|
|
||||||
|
videobin->user_video_encoder = encoder;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_video_recording_bin_set_property (GObject * object, guint prop_id,
|
gst_video_recording_bin_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec)
|
const GValue * value, GParamSpec * pspec)
|
||||||
|
@ -73,11 +99,18 @@ gst_video_recording_bin_set_property (GObject * object, guint prop_id,
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_LOCATION:
|
case PROP_LOCATION:
|
||||||
|
if (videobin->location)
|
||||||
|
g_free (videobin->location);
|
||||||
|
|
||||||
videobin->location = g_value_dup_string (value);
|
videobin->location = g_value_dup_string (value);
|
||||||
if (videobin->sink) {
|
if (videobin->sink) {
|
||||||
g_object_set (videobin->sink, "location", videobin->location, NULL);
|
g_object_set (videobin->sink, "location", videobin->location, NULL);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case PROP_VIDEO_ENCODER:
|
||||||
|
gst_video_recording_bin_set_video_encoder (videobin,
|
||||||
|
g_value_get_object (value));
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
break;
|
break;
|
||||||
|
@ -94,6 +127,9 @@ gst_video_recording_bin_get_property (GObject * object, guint prop_id,
|
||||||
case PROP_LOCATION:
|
case PROP_LOCATION:
|
||||||
g_value_set_string (value, videobin->location);
|
g_value_set_string (value, videobin->location);
|
||||||
break;
|
break;
|
||||||
|
case PROP_VIDEO_ENCODER:
|
||||||
|
g_value_set_object (value, videobin->video_encoder);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
break;
|
break;
|
||||||
|
@ -122,6 +158,9 @@ gst_video_recording_bin_class_init (GstVideoRecordingBinClass * klass)
|
||||||
gobject_class = G_OBJECT_CLASS (klass);
|
gobject_class = G_OBJECT_CLASS (klass);
|
||||||
element_class = GST_ELEMENT_CLASS (klass);
|
element_class = GST_ELEMENT_CLASS (klass);
|
||||||
|
|
||||||
|
gobject_class->dispose = gst_video_recording_bin_dispose;
|
||||||
|
gobject_class->finalize = gst_video_recording_bin_finalize;
|
||||||
|
|
||||||
gobject_class->set_property = gst_video_recording_bin_set_property;
|
gobject_class->set_property = gst_video_recording_bin_set_property;
|
||||||
gobject_class->get_property = gst_video_recording_bin_get_property;
|
gobject_class->get_property = gst_video_recording_bin_get_property;
|
||||||
|
|
||||||
|
@ -132,6 +171,11 @@ gst_video_recording_bin_class_init (GstVideoRecordingBinClass * klass)
|
||||||
g_param_spec_string ("location", "Location",
|
g_param_spec_string ("location", "Location",
|
||||||
"Location to save the captured files.",
|
"Location to save the captured files.",
|
||||||
DEFAULT_LOCATION, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
DEFAULT_LOCATION, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||||
|
|
||||||
|
g_object_class_install_property (gobject_class, PROP_VIDEO_ENCODER,
|
||||||
|
g_param_spec_object ("video-encoder", "Video encoder",
|
||||||
|
"Video encoder GstElement (default is theoraenc).",
|
||||||
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -144,13 +188,39 @@ gst_video_recording_bin_init (GstVideoRecordingBin * videobin,
|
||||||
gst_element_add_pad (GST_ELEMENT_CAST (videobin), videobin->ghostpad);
|
gst_element_add_pad (GST_ELEMENT_CAST (videobin), videobin->ghostpad);
|
||||||
|
|
||||||
videobin->location = g_strdup (DEFAULT_LOCATION);
|
videobin->location = g_strdup (DEFAULT_LOCATION);
|
||||||
|
videobin->video_encoder = NULL;
|
||||||
|
videobin->user_video_encoder = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_video_recording_bin_dispose (GObject * object)
|
||||||
|
{
|
||||||
|
GstVideoRecordingBin *videobin = GST_VIDEO_RECORDING_BIN_CAST (object);
|
||||||
|
|
||||||
|
if (videobin->user_video_encoder) {
|
||||||
|
gst_object_unref (videobin->user_video_encoder);
|
||||||
|
videobin->user_video_encoder = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (parent_class)->dispose ((GObject *) videobin);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_video_recording_bin_finalize (GObject * object)
|
||||||
|
{
|
||||||
|
GstVideoRecordingBin *videobin = GST_VIDEO_RECORDING_BIN_CAST (object);
|
||||||
|
|
||||||
|
g_free (videobin->location);
|
||||||
|
videobin->location = NULL;
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (parent_class)->finalize ((GObject *) videobin);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_video_recording_bin_create_elements (GstVideoRecordingBin * videobin)
|
gst_video_recording_bin_create_elements (GstVideoRecordingBin * videobin)
|
||||||
{
|
{
|
||||||
GstElement *colorspace;
|
GstElement *colorspace;
|
||||||
GstElement *encoder;
|
|
||||||
GstElement *muxer;
|
GstElement *muxer;
|
||||||
GstElement *sink;
|
GstElement *sink;
|
||||||
GstPad *pad = NULL;
|
GstPad *pad = NULL;
|
||||||
|
@ -160,31 +230,38 @@ gst_video_recording_bin_create_elements (GstVideoRecordingBin * videobin)
|
||||||
|
|
||||||
/* create elements */
|
/* create elements */
|
||||||
colorspace =
|
colorspace =
|
||||||
gst_element_factory_make ("ffmpegcolorspace", "videobin-colorspace");
|
gst_camerabin_create_and_add_element (GST_BIN (videobin),
|
||||||
|
DEFAULT_COLORSPACE);
|
||||||
if (!colorspace)
|
if (!colorspace)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
encoder = gst_element_factory_make ("theoraenc", "videobin-encoder");
|
if (videobin->user_video_encoder) {
|
||||||
if (!encoder)
|
videobin->video_encoder = videobin->user_video_encoder;
|
||||||
goto error;
|
if (!gst_camerabin_add_element (GST_BIN (videobin),
|
||||||
|
videobin->video_encoder)) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
videobin->video_encoder =
|
||||||
|
gst_camerabin_create_and_add_element (GST_BIN (videobin),
|
||||||
|
DEFAULT_VIDEO_ENCODER);
|
||||||
|
if (!videobin->video_encoder)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
muxer = gst_element_factory_make ("oggmux", "videobin->muxer");
|
muxer = gst_camerabin_create_and_add_element (GST_BIN (videobin),
|
||||||
|
DEFAULT_MUXER);
|
||||||
if (!muxer)
|
if (!muxer)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
sink = gst_element_factory_make ("filesink", "videobin-sink");
|
sink = gst_camerabin_create_and_add_element (GST_BIN (videobin),
|
||||||
|
DEFAULT_SINK);
|
||||||
if (!sink)
|
if (!sink)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
videobin->sink = gst_object_ref (sink);
|
videobin->sink = gst_object_ref (sink);
|
||||||
g_object_set (sink, "location", videobin->location, "async", FALSE, NULL);
|
g_object_set (sink, "location", videobin->location, "async", FALSE, NULL);
|
||||||
|
|
||||||
/* add and link */
|
|
||||||
gst_bin_add_many (GST_BIN_CAST (videobin), colorspace, encoder, muxer, sink,
|
|
||||||
NULL);
|
|
||||||
if (!gst_element_link_many (colorspace, encoder, muxer, sink, NULL))
|
|
||||||
goto error;
|
|
||||||
|
|
||||||
/* add ghostpad */
|
/* add ghostpad */
|
||||||
pad = gst_element_get_static_pad (colorspace, "sink");
|
pad = gst_element_get_static_pad (colorspace, "sink");
|
||||||
if (!gst_ghost_pad_set_target (GST_GHOST_PAD (videobin->ghostpad), pad))
|
if (!gst_ghost_pad_set_target (GST_GHOST_PAD (videobin->ghostpad), pad))
|
||||||
|
|
|
@ -42,6 +42,8 @@ struct _GstVideoRecordingBin
|
||||||
|
|
||||||
/* props */
|
/* props */
|
||||||
gchar *location;
|
gchar *location;
|
||||||
|
GstElement *video_encoder;
|
||||||
|
GstElement *user_video_encoder;
|
||||||
|
|
||||||
gboolean elements_created;
|
gboolean elements_created;
|
||||||
};
|
};
|
||||||
|
|
|
@ -110,6 +110,50 @@ GST_START_TEST (test_simple_recording)
|
||||||
|
|
||||||
GST_END_TEST;
|
GST_END_TEST;
|
||||||
|
|
||||||
|
GST_START_TEST (test_setting_video_encoder)
|
||||||
|
{
|
||||||
|
GstVideoRecordingBinTestContext ctx;
|
||||||
|
GstBus *bus;
|
||||||
|
GstMessage *msg;
|
||||||
|
GstElement *encoder;
|
||||||
|
gchar *test_file_name;
|
||||||
|
FILE *f;
|
||||||
|
|
||||||
|
gstvideorecordingbin_init_test_context (&ctx, N_BUFFERS);
|
||||||
|
bus = gst_element_get_bus (ctx.pipe);
|
||||||
|
|
||||||
|
test_file_name = make_test_file_name (0);
|
||||||
|
g_object_set (ctx.vrbin, "location", test_file_name, NULL);
|
||||||
|
|
||||||
|
encoder = gst_element_factory_make ("theoraenc", NULL);
|
||||||
|
g_object_set (ctx.vrbin, "video-encoder", encoder, NULL);
|
||||||
|
|
||||||
|
fail_if (gst_element_set_state (ctx.pipe, GST_STATE_PLAYING) ==
|
||||||
|
GST_STATE_CHANGE_FAILURE);
|
||||||
|
|
||||||
|
msg = gst_bus_timed_pop_filtered (bus, GST_SECOND * 10,
|
||||||
|
GST_MESSAGE_EOS | GST_MESSAGE_ERROR);
|
||||||
|
fail_unless (msg != NULL);
|
||||||
|
fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
|
||||||
|
|
||||||
|
/* check there is a recorded file */
|
||||||
|
fail_unless (g_file_test (test_file_name, G_FILE_TEST_EXISTS));
|
||||||
|
fail_unless (g_file_test (test_file_name, G_FILE_TEST_IS_REGULAR));
|
||||||
|
fail_if (g_file_test (test_file_name, G_FILE_TEST_IS_SYMLINK));
|
||||||
|
|
||||||
|
/* check the file isn't empty */
|
||||||
|
f = fopen (test_file_name, "r");
|
||||||
|
fseek (f, 0, SEEK_END);
|
||||||
|
fail_unless (ftell (f) > 0);
|
||||||
|
fclose (f);
|
||||||
|
|
||||||
|
gstvideorecordingbin_unset_test_context (&ctx);
|
||||||
|
gst_object_unref (bus);
|
||||||
|
g_free (test_file_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_END_TEST;
|
||||||
|
|
||||||
static Suite *
|
static Suite *
|
||||||
videorecordingbin_suite (void)
|
videorecordingbin_suite (void)
|
||||||
{
|
{
|
||||||
|
@ -118,6 +162,7 @@ videorecordingbin_suite (void)
|
||||||
|
|
||||||
suite_add_tcase (s, tc_chain);
|
suite_add_tcase (s, tc_chain);
|
||||||
tcase_add_test (tc_chain, test_simple_recording);
|
tcase_add_test (tc_chain, test_simple_recording);
|
||||||
|
tcase_add_test (tc_chain, test_setting_video_encoder);
|
||||||
|
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue