mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-22 16:26:39 +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.
238 lines
7 KiB
C
238 lines
7 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-giosrc
|
|
* @short_description: Read from any GIO-supported location
|
|
* @see_also: #GstFileSrc, #GstGnomeVFSSrc, #GstGioSink
|
|
*
|
|
* <refsect2>
|
|
* <para>
|
|
* This plugin reads data from 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', 'http',
|
|
* 'ftp', or 'smb'.
|
|
* </para>
|
|
* <para>
|
|
* Example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v giosrc location=file:///home/joe/foo.xyz ! fakesink
|
|
* </programlisting>
|
|
* The above pipeline will simply read a local file and do nothing with the
|
|
* data read. Instead of giosrc, we could just as well have used the
|
|
* filesrc element here.
|
|
* </para>
|
|
* <para>
|
|
* Another example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v giosrc location=smb://othercomputer/foo.xyz ! filesink location=/home/joe/foo.xyz
|
|
* </programlisting>
|
|
* The above pipeline will copy a file from a remote host to the local file
|
|
* system using the Samba protocol.
|
|
* </para>
|
|
* <para>
|
|
* Yet another example pipeline:
|
|
* <programlisting>
|
|
* gst-launch -v giosrc location=http://music.foobar.com/demo.mp3 ! mad ! audioconvert ! audioresample ! alsasink
|
|
* </programlisting>
|
|
* The above pipeline will read and decode and play an mp3 file from a
|
|
* web server using the http protocol.
|
|
* </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 "gstgiosrc.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_gio_src_debug);
|
|
#define GST_CAT_DEFAULT gst_gio_src_debug
|
|
|
|
enum
|
|
{
|
|
ARG_0,
|
|
ARG_LOCATION
|
|
};
|
|
|
|
GST_BOILERPLATE_FULL (GstGioSrc, gst_gio_src, GstGioBaseSrc,
|
|
GST_TYPE_GIO_BASE_SRC, gst_gio_uri_handler_do_init);
|
|
|
|
static void gst_gio_src_finalize (GObject * object);
|
|
static void gst_gio_src_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec);
|
|
static void gst_gio_src_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
static gboolean gst_gio_src_start (GstBaseSrc * base_src);
|
|
|
|
static void
|
|
gst_gio_src_base_init (gpointer gclass)
|
|
{
|
|
static GstElementDetails element_details = {
|
|
"GIO source",
|
|
"Source/File",
|
|
"Read from 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_src_debug, "gio_src", 0, "GIO source");
|
|
|
|
gst_element_class_set_details (element_class, &element_details);
|
|
}
|
|
|
|
static void
|
|
gst_gio_src_class_init (GstGioSrcClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GstElementClass *gstelement_class;
|
|
GstBaseSrcClass *gstbasesrc_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
gstelement_class = (GstElementClass *) klass;
|
|
gstbasesrc_class = (GstBaseSrcClass *) klass;
|
|
|
|
gobject_class->finalize = gst_gio_src_finalize;
|
|
gobject_class->set_property = gst_gio_src_set_property;
|
|
gobject_class->get_property = gst_gio_src_get_property;
|
|
|
|
g_object_class_install_property (gobject_class, ARG_LOCATION,
|
|
g_param_spec_string ("location", "Location", "URI location to read from",
|
|
NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
gstbasesrc_class->start = GST_DEBUG_FUNCPTR (gst_gio_src_start);
|
|
}
|
|
|
|
static void
|
|
gst_gio_src_init (GstGioSrc * src, GstGioSrcClass * gclass)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gst_gio_src_finalize (GObject * object)
|
|
{
|
|
GstGioSrc *src = GST_GIO_SRC (object);
|
|
|
|
if (src->location) {
|
|
g_free (src->location);
|
|
src->location = NULL;
|
|
}
|
|
|
|
GST_CALL_PARENT (G_OBJECT_CLASS, finalize, (object));
|
|
}
|
|
|
|
static void
|
|
gst_gio_src_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstGioSrc *src = GST_GIO_SRC (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_LOCATION:
|
|
if (GST_STATE (src) == GST_STATE_PLAYING ||
|
|
GST_STATE (src) == GST_STATE_PAUSED)
|
|
break;
|
|
|
|
g_free (src->location);
|
|
src->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_src_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstGioSrc *src = GST_GIO_SRC (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_LOCATION:
|
|
g_value_set_string (value, src->location);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_gio_src_start (GstBaseSrc * base_src)
|
|
{
|
|
GstGioSrc *src = GST_GIO_SRC (base_src);
|
|
GFile *file;
|
|
GError *err = NULL;
|
|
GInputStream *stream;
|
|
GCancellable *cancel = GST_GIO_BASE_SRC (src)->cancel;
|
|
|
|
if (src->location == NULL) {
|
|
GST_ELEMENT_ERROR (src, RESOURCE, OPEN_READ, (NULL), ("No location given"));
|
|
return FALSE;
|
|
}
|
|
|
|
file = g_file_new_for_uri (src->location);
|
|
|
|
if (file == NULL) {
|
|
GST_ELEMENT_ERROR (src, RESOURCE, OPEN_READ, (NULL),
|
|
("Malformed URI or protocol not supported (%s)", src->location));
|
|
return FALSE;
|
|
}
|
|
|
|
stream = G_INPUT_STREAM (g_file_read (file, cancel, &err));
|
|
|
|
g_object_unref (file);
|
|
|
|
if (stream == NULL && !gst_gio_error (src, "g_file_read", &err, NULL)) {
|
|
|
|
if (GST_GIO_ERROR_MATCHES (err, NOT_FOUND)) {
|
|
GST_ELEMENT_ERROR (src, RESOURCE, NOT_FOUND, (NULL),
|
|
("Could not open location %s for reading: %s",
|
|
src->location, err->message));
|
|
} else {
|
|
GST_ELEMENT_ERROR (src, RESOURCE, OPEN_READ, (NULL),
|
|
("Could not open location %s for reading: %s",
|
|
src->location, err->message));
|
|
}
|
|
|
|
g_clear_error (&err);
|
|
return FALSE;
|
|
} else if (stream == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
gst_gio_base_src_set_stream (GST_GIO_BASE_SRC (src), stream);
|
|
|
|
GST_DEBUG_OBJECT (src, "opened location %s", src->location);
|
|
|
|
return GST_BASE_SRC_CLASS (parent_class)->start (base_src);
|
|
}
|