mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 21:18:52 +00:00
49deb0c05d
Original commit message from CVS: * configure.ac: * ext/alsa/gstalsamixerelement.c: (gst_alsa_mixer_element_class_init): * ext/alsa/gstalsasink.c: (gst_alsasink_class_init): * ext/alsa/gstalsasrc.c: (gst_alsasrc_class_init): * ext/cdparanoia/gstcdparanoiasrc.c: (gst_cd_paranoia_src_class_init): * ext/gio/gstgiosink.c: (gst_gio_sink_class_init): * ext/gio/gstgiosrc.c: (gst_gio_src_class_init): * ext/gio/gstgiostreamsink.c: (gst_gio_stream_sink_class_init): * ext/gio/gstgiostreamsrc.c: (gst_gio_stream_src_class_init): * ext/gnomevfs/gstgnomevfssink.c: (gst_gnome_vfs_sink_class_init): * ext/gnomevfs/gstgnomevfssrc.c: (gst_gnome_vfs_src_class_init): * ext/ogg/gstoggmux.c: (gst_ogg_mux_class_init): * ext/pango/gsttextoverlay.c: (gst_text_overlay_class_init): * ext/pango/gsttextrender.c: (gst_text_render_class_init): * ext/theora/theoradec.c: (gst_theora_dec_class_init): * ext/theora/theoraenc.c: (gst_theora_enc_class_init): * ext/theora/theoraparse.c: (gst_theora_parse_class_init): * ext/vorbis/vorbisenc.c: (gst_vorbis_enc_class_init): * gst-libs/gst/audio/gstaudiofiltertemplate.c: (gst_audio_filter_template_class_init): * gst-libs/gst/audio/gstbaseaudiosink.c: (gst_base_audio_sink_class_init): * gst-libs/gst/audio/gstbaseaudiosrc.c: (gst_base_audio_src_class_init): * gst-libs/gst/cdda/gstcddabasesrc.c: (gst_cdda_base_src_class_init): * gst-libs/gst/interfaces/mixertrack.c: (gst_mixer_track_class_init): * gst-libs/gst/rtp/gstbasertpdepayload.c: (gst_base_rtp_depayload_class_init): * gst-libs/gst/rtp/gstbasertppayload.c: (gst_basertppayload_class_init): * gst/audioconvert/gstaudioconvert.c: (gst_audio_convert_class_init): * gst/audiorate/gstaudiorate.c: (gst_audio_rate_class_init): * gst/audioresample/gstaudioresample.c: (gst_audioresample_class_init): * gst/audiotestsrc/gstaudiotestsrc.c: (gst_audio_test_src_class_init): * gst/gdp/gstgdppay.c: (gst_gdp_pay_class_init): * gst/playback/gstdecodebin2.c: (gst_decode_bin_class_init): * gst/playback/gstplaybasebin.c: (gst_play_base_bin_class_init), (preroll_unlinked): * gst/playback/gstplaybin.c: (gst_play_bin_class_init): * gst/playback/gstplaybin2.c: (gst_play_bin_class_init): * gst/playback/gstplaysink.c: (gst_play_sink_class_init): * gst/playback/gstqueue2.c: (gst_queue_class_init): * gst/playback/gststreaminfo.c: (gst_stream_info_class_init): * gst/playback/gststreamselector.c: (gst_selector_pad_class_init), (gst_stream_selector_class_init): * gst/playback/gsturidecodebin.c: (gst_uri_decode_bin_class_init): * gst/subparse/gstsubparse.c: (gst_sub_parse_class_init): * gst/tcp/gstmultifdsink.c: (gst_multi_fd_sink_class_init): * gst/tcp/gsttcpclientsink.c: (gst_tcp_client_sink_class_init): * gst/tcp/gsttcpclientsrc.c: (gst_tcp_client_src_class_init): * gst/tcp/gsttcpserversink.c: (gst_tcp_server_sink_class_init): * gst/tcp/gsttcpserversrc.c: (gst_tcp_server_src_class_init): * gst/videorate/gstvideorate.c: (gst_video_rate_class_init): * gst/videoscale/gstvideoscale.c: (gst_video_scale_class_init): * gst/videotestsrc/gstvideotestsrc.c: (gst_video_test_src_class_init): * gst/volume/gstvolume.c: (gst_volume_class_init): * sys/v4l/gstv4lelement.c: (gst_v4lelement_class_init): * sys/v4l/gstv4lmjpegsink.c: (gst_v4lmjpegsink_class_init): * sys/v4l/gstv4lmjpegsrc.c: (gst_v4lmjpegsrc_class_init): * sys/v4l/gstv4lsrc.c: (gst_v4lsrc_class_init): * sys/ximage/ximagesink.c: (gst_ximagesink_class_init): * sys/xvimage/xvimagesink.c: (gst_xvimagesink_class_init): Use G_PARAM_STATIC_STRINGS everywhere for GParamSpecs that use static strings (i.e. all). This gives us less memory usage, fewer allocations and thus less memory defragmentation. Depend on core CVS for this. Fixes bug #523806.
250 lines
7.3 KiB
C
250 lines
7.3 KiB
C
/* GStreamer
|
|
*
|
|
* Copyright (C) 2007 Rene Stadler <mail@renestadler.de>
|
|
* Copyright (C) 2007 Sebastian Dröge <slomo@circular-chaos.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:element-giosink
|
|
* @short_description: Write a stream to any GIO-supported location
|
|
* @see_also: #GstFileSink, #GstGnomeVFSSink, #GstGioSrc
|
|
*
|
|
* <refsect2>
|
|
* <para>
|
|
* This plugin writes incoming data to a local or remote location specified
|
|
* by an URI. This location can be specified using any protocol supported by
|
|
* the GIO library or it's VFS backends. Common protocols are 'file', 'ftp',
|
|
* or 'smb'.
|
|
* </para>
|
|
* <para>
|
|
* Example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v filesrc location=input.xyz ! giosink location=file:///home/joe/out.xyz
|
|
* </programlisting>
|
|
* The above pipeline will simply copy a local file. Instead of giosink,
|
|
* we could just as well have used the filesink element here.
|
|
* </para>
|
|
* <para>
|
|
* Another example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v filesrc location=foo.mp3 ! mad ! flacenc ! giosink location=smb://othercomputer/foo.flac
|
|
* </programlisting>
|
|
* The above pipeline will re-encode an mp3 file into FLAC format and store
|
|
* it on a remote host using the Samba protocol.
|
|
* </para>
|
|
* <para>
|
|
* Another example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v audiotestsrc num-buffers=100 ! vorbisenc ! oggmux ! giosink location=file:///home/foo/bar.ogg
|
|
* </programlisting>
|
|
* The above pipeline will encode a 440Hz sine wave to Ogg Vorbis and stores
|
|
* it in the home directory of user foo.
|
|
* </para>
|
|
* </refsect2>
|
|
*/
|
|
|
|
/* FIXME: We would like to mount the enclosing volume of an URL
|
|
* if it isn't mounted yet but this is possible async-only.
|
|
* Unfortunately this requires a running main loop from the
|
|
* default context and we can't guarantuee this!
|
|
*
|
|
* We would also like to do authentication while mounting.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "gstgiosink.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_gio_sink_debug);
|
|
#define GST_CAT_DEFAULT gst_gio_sink_debug
|
|
|
|
/* Filter signals and args */
|
|
enum
|
|
{
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
ARG_0,
|
|
ARG_LOCATION
|
|
};
|
|
|
|
GST_BOILERPLATE_FULL (GstGioSink, gst_gio_sink, GstGioBaseSink,
|
|
GST_TYPE_GIO_BASE_SINK, gst_gio_uri_handler_do_init);
|
|
|
|
static void gst_gio_sink_finalize (GObject * object);
|
|
static void gst_gio_sink_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec);
|
|
static void gst_gio_sink_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
static gboolean gst_gio_sink_start (GstBaseSink * base_sink);
|
|
|
|
static void
|
|
gst_gio_sink_base_init (gpointer gclass)
|
|
{
|
|
static GstElementDetails element_details = {
|
|
"GIO sink",
|
|
"Sink/File",
|
|
"Write to any GIO-supported location",
|
|
"Ren\xc3\xa9 Stadler <mail@renestadler.de>, "
|
|
"Sebastian Dröge <slomo@circular-chaos.org>"
|
|
};
|
|
GstElementClass *element_class = GST_ELEMENT_CLASS (gclass);
|
|
|
|
GST_DEBUG_CATEGORY_INIT (gst_gio_sink_debug, "gio_sink", 0, "GIO sink");
|
|
|
|
gst_element_class_set_details (element_class, &element_details);
|
|
}
|
|
|
|
static void
|
|
gst_gio_sink_class_init (GstGioSinkClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GstElementClass *gstelement_class;
|
|
GstBaseSinkClass *gstbasesink_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
gstelement_class = (GstElementClass *) klass;
|
|
gstbasesink_class = (GstBaseSinkClass *) klass;
|
|
|
|
gobject_class->finalize = gst_gio_sink_finalize;
|
|
gobject_class->set_property = gst_gio_sink_set_property;
|
|
gobject_class->get_property = gst_gio_sink_get_property;
|
|
|
|
g_object_class_install_property (gobject_class, ARG_LOCATION,
|
|
g_param_spec_string ("location", "Location", "URI location to write to",
|
|
NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
gstbasesink_class->start = GST_DEBUG_FUNCPTR (gst_gio_sink_start);
|
|
}
|
|
|
|
static void
|
|
gst_gio_sink_init (GstGioSink * sink, GstGioSinkClass * gclass)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gst_gio_sink_finalize (GObject * object)
|
|
{
|
|
GstGioSink *sink = GST_GIO_SINK (object);
|
|
|
|
if (sink->location) {
|
|
g_free (sink->location);
|
|
sink->location = NULL;
|
|
}
|
|
|
|
GST_CALL_PARENT (G_OBJECT_CLASS, finalize, (object));
|
|
}
|
|
|
|
static void
|
|
gst_gio_sink_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstGioSink *sink = GST_GIO_SINK (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_LOCATION:
|
|
if (GST_STATE (sink) == GST_STATE_PLAYING ||
|
|
GST_STATE (sink) == GST_STATE_PAUSED)
|
|
break;
|
|
|
|
g_free (sink->location);
|
|
sink->location = g_strdup (g_value_get_string (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_gio_sink_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstGioSink *sink = GST_GIO_SINK (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_LOCATION:
|
|
g_value_set_string (value, sink->location);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_gio_sink_start (GstBaseSink * base_sink)
|
|
{
|
|
GstGioSink *sink = GST_GIO_SINK (base_sink);
|
|
GFile *file;
|
|
GOutputStream *stream;
|
|
GCancellable *cancel = GST_GIO_BASE_SINK (sink)->cancel;
|
|
gboolean success;
|
|
GError *err = NULL;
|
|
|
|
if (sink->location == NULL) {
|
|
GST_ELEMENT_ERROR (sink, RESOURCE, OPEN_WRITE, (NULL),
|
|
("No location given"));
|
|
return FALSE;
|
|
}
|
|
|
|
file = g_file_new_for_uri (sink->location);
|
|
|
|
if (file == NULL) {
|
|
GST_ELEMENT_ERROR (sink, RESOURCE, OPEN_WRITE, (NULL),
|
|
("Malformed URI or protocol not supported (%s)", sink->location));
|
|
return FALSE;
|
|
}
|
|
|
|
stream =
|
|
G_OUTPUT_STREAM (g_file_create (file, G_FILE_CREATE_NONE, cancel, &err));
|
|
|
|
success = (stream != NULL);
|
|
|
|
g_object_unref (file);
|
|
|
|
if (!success && !gst_gio_error (sink, "g_file_create", &err, NULL)) {
|
|
|
|
/*if (GST_GIO_ERROR_MATCHES (err, EXISTS)) */
|
|
/* FIXME: Retry with replace if overwrite == TRUE! */
|
|
|
|
if (GST_GIO_ERROR_MATCHES (err, NOT_FOUND))
|
|
GST_ELEMENT_ERROR (sink, RESOURCE, NOT_FOUND, (NULL),
|
|
("Could not open location %s for writing: %s",
|
|
sink->location, err->message));
|
|
else
|
|
GST_ELEMENT_ERROR (sink, RESOURCE, OPEN_READ, (NULL),
|
|
("Could not open location %s for writing: %s",
|
|
sink->location, err->message));
|
|
|
|
g_clear_error (&err);
|
|
}
|
|
|
|
if (!success)
|
|
return FALSE;
|
|
|
|
GST_DEBUG_OBJECT (sink, "opened location %s", sink->location);
|
|
|
|
gst_gio_base_sink_set_stream (GST_GIO_BASE_SINK (sink), stream);
|
|
|
|
return GST_BASE_SINK_CLASS (parent_class)->start (base_sink);
|
|
}
|