summaryrefslogtreecommitdiff
path: root/farstream/fs-stream-transmitter.c
diff options
context:
space:
mode:
authorOlivier CrĂȘte <olivier.crete@collabora.com>2011-10-11 16:08:49 -0400
committerOlivier CrĂȘte <olivier.crete@collabora.com>2011-10-11 16:14:10 -0400
commit52a59229d200a8d74d66e02126c290434d3157d8 (patch)
tree1adf3344ad47dbed6c35c42511362b232c703683 /farstream/fs-stream-transmitter.c
parent7098f40db04b6a32c311b802a2c12f0f450ee7b7 (diff)
downloadfarstream-52a59229d200a8d74d66e02126c290434d3157d8.tar.gz
Move the lib out of gst-libs
Diffstat (limited to 'farstream/fs-stream-transmitter.c')
-rw-r--r--farstream/fs-stream-transmitter.c453
1 files changed, 453 insertions, 0 deletions
diff --git a/farstream/fs-stream-transmitter.c b/farstream/fs-stream-transmitter.c
new file mode 100644
index 00000000..979a2eb2
--- /dev/null
+++ b/farstream/fs-stream-transmitter.c
@@ -0,0 +1,453 @@
+/*
+ * Farstream - Farstream Stream Transmitter
+ *
+ * Copyright 2007 Collabora Ltd.
+ * @author: Olivier Crete <olivier.crete@collabora.co.uk>
+ * Copyright 2007 Nokia Corp.
+ *
+ * fs-stream-transmitter.c - A Farstream Stream Transmitter gobject
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+/**
+ * SECTION:fs-stream-transmitter
+ * @short_description: A stream transmitter object used to convey per-stream
+ * information to a transmitter.
+ *
+ * This object is the base implementation of a Farstream Stream Transmitter.
+ * It needs to be derived and implement by a Farstream transmitter.
+ * A Farstream Stream transmitter is used to convery per-stream information
+ * to a transmitter, this is mostly local and remote candidates
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include "fs-stream-transmitter.h"
+
+#include <gst/gst.h>
+
+#include "fs-marshal.h"
+#include "fs-conference.h"
+#include "fs-private.h"
+
+#define GST_CAT_DEFAULT fs_conference_debug
+
+/* Signals */
+enum
+{
+ ERROR_SIGNAL,
+ NEW_LOCAL_CANDIDATE,
+ NEW_ACTIVE_CANDIDATE_PAIR,
+ LOCAL_CANDIDATES_PREPARED,
+ KNOWN_SOURCE_PACKET_RECEIVED,
+ STATE_CHANGED,
+ LAST_SIGNAL
+};
+
+/* props */
+enum
+{
+ PROP_0,
+ PROP_SENDING,
+ PROP_PREFERRED_LOCAL_CANDIDATES,
+ PROP_ASSOCIATE_ON_SOURCE
+};
+
+struct _FsStreamTransmitterPrivate
+{
+ gboolean disposed;
+};
+
+G_DEFINE_ABSTRACT_TYPE(FsStreamTransmitter, fs_stream_transmitter,
+ GST_TYPE_OBJECT);
+
+
+#define FS_STREAM_TRANSMITTER_GET_PRIVATE(o) \
+ (G_TYPE_INSTANCE_GET_PRIVATE ((o), FS_TYPE_STREAM_TRANSMITTER, \
+ FsStreamTransmitterPrivate))
+
+static void fs_stream_transmitter_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec);
+static void fs_stream_transmitter_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec);
+
+static guint signals[LAST_SIGNAL] = { 0 };
+
+static void
+fs_stream_transmitter_class_init (FsStreamTransmitterClass *klass)
+{
+ GObjectClass *gobject_class;
+
+ gobject_class = (GObjectClass *) klass;
+
+ gobject_class->set_property = fs_stream_transmitter_set_property;
+ gobject_class->get_property = fs_stream_transmitter_get_property;
+
+
+ /**
+ * FsStreamTransmitter:sending:
+ *
+ * A network source #GstElement to be used by the #FsSession
+ *
+ */
+ g_object_class_install_property (gobject_class,
+ PROP_SENDING,
+ g_param_spec_boolean ("sending",
+ "Whether to send from this transmitter",
+ "If set to FALSE, the transmitter will stop sending to this person",
+ TRUE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+
+ /**
+ * FsStreamTransmitter:preferred-local-candidate:
+ *
+ * The list of preferred local candidates for this stream
+ * It is a #GList of #FsCandidates
+ *
+ */
+ g_object_class_install_property (gobject_class,
+ PROP_PREFERRED_LOCAL_CANDIDATES,
+ g_param_spec_boxed ("preferred-local-candidates",
+ "The preferred candidates",
+ "A GList of FsCandidates",
+ FS_TYPE_CANDIDATE_LIST,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+
+ /**
+ * FsStreamTransmitter:associate-on-source
+ *
+ * This tells the stream transmitter to associate incoming data with this
+ * based on the source without looking at the content if possible.
+ *
+ */
+
+ g_object_class_install_property (gobject_class,
+ PROP_ASSOCIATE_ON_SOURCE,
+ g_param_spec_boolean ("associate-on-source",
+ "Associate incoming data based on the source address",
+ "Whether to associate incoming data stream based on the source address",
+ TRUE,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+
+ /**
+ * FsStreamTransmitter::error:
+ * @self: #FsStreamTransmitter that emitted the signal
+ * @errorno: The number of the error
+ * @error_msg: Error message (for the programmer)
+ *
+ * This signal is emitted in any error condition
+ *
+ */
+ signals[ERROR_SIGNAL] = g_signal_new ("error",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ _fs_marshal_VOID__ENUM_STRING,
+ G_TYPE_NONE, 2, FS_TYPE_ERROR, G_TYPE_STRING);
+
+ /**
+ * FsStreamTransmitter::new-active-candidate-pair:
+ * @self: #FsStreamTransmitter that emitted the signal
+ * @local_candidate: #FsCandidate of the local candidate being used
+ * @remote_candidate: #FsCandidate of the remote candidate being used
+ *
+ * This signal is emitted when there is a new active chandidate pair that has
+ * been established. This is specially useful for ICE where the active
+ * candidate pair can change automatically due to network conditions. The user
+ * must not modify the candidates and must copy them if he wants to use them
+ * outside the callback scope.
+ *
+ */
+ signals[NEW_ACTIVE_CANDIDATE_PAIR] = g_signal_new
+ ("new-active-candidate-pair",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ _fs_marshal_VOID__BOXED_BOXED,
+ G_TYPE_NONE, 2, FS_TYPE_CANDIDATE, FS_TYPE_CANDIDATE);
+
+ /**
+ * FsStreamTransmitter::new-local-candidate:
+ * @self: #FsStream that emitted the signal
+ * @local_candidate: #FsCandidate of the local candidate
+ *
+ * This signal is emitted when a new local candidate is discovered.
+ *
+ */
+ signals[NEW_LOCAL_CANDIDATE] = g_signal_new
+ ("new-local-candidate",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ g_cclosure_marshal_VOID__BOXED,
+ G_TYPE_NONE, 1, FS_TYPE_CANDIDATE);
+
+ /**
+ * FsStreamTransmitter::local-candidates-prepared:
+ * @self: #FsStreamTransmitter that emitted the signal
+ *
+ * This signal is emitted when all local candidates have been
+ * prepared, an ICE implementation would send its SDP offer or answer.
+ *
+ */
+ signals[LOCAL_CANDIDATES_PREPARED] = g_signal_new
+ ("local-candidates-prepared",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE, 0);
+
+ /**
+ * FsStreamTransmitter::known-source-packet-received:
+ * @self: #FsStreamTransmitter that emitted the signal
+ * @component: The Component on which this buffer was received
+ * @buffer: the #GstBuffer coming from the known source
+ *
+ * This signal is emitted when a buffer coming from a confirmed known source
+ * is received.
+ */
+ signals[KNOWN_SOURCE_PACKET_RECEIVED] = g_signal_new
+ ("known-source-packet-received",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ _fs_marshal_VOID__UINT_POINTER,
+ G_TYPE_NONE, 2, G_TYPE_UINT, G_TYPE_POINTER);
+
+
+ /**
+ * FsStreamTransmitter::state-changed
+ * @self: #FsStreamTransmitter that emitted the signal
+ * @component: the id of the component which state has changed
+ * @state: the new state of the component
+ *
+ * This signal is emitted when the ICE state (or equivalent) of the component
+ * changes
+ */
+ signals[STATE_CHANGED] = g_signal_new
+ ("state-changed",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL,
+ NULL,
+ _fs_marshal_VOID__UINT_ENUM,
+ G_TYPE_NONE, 2, G_TYPE_UINT, FS_TYPE_STREAM_STATE);
+
+
+ g_type_class_add_private (klass, sizeof (FsStreamTransmitterPrivate));
+}
+
+static void
+fs_stream_transmitter_init (FsStreamTransmitter *self)
+{
+ /* member init */
+ self->priv = FS_STREAM_TRANSMITTER_GET_PRIVATE (self);
+ self->priv->disposed = FALSE;
+}
+
+static void
+fs_stream_transmitter_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GST_WARNING ("Subclass %s of FsStreamTransmitter does not override the %s"
+ " property getter",
+ G_OBJECT_TYPE_NAME(object),
+ g_param_spec_get_name (pspec));
+}
+
+static void
+fs_stream_transmitter_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ switch (prop_id)
+ {
+ /* These properties, we can safely not override */
+ case PROP_ASSOCIATE_ON_SOURCE:
+ break;
+ default:
+ GST_WARNING ("Subclass %s of FsStreamTransmitter does not override the %s"
+ " property setter",
+ G_OBJECT_TYPE_NAME(object),
+ g_param_spec_get_name (pspec));
+ break;
+ }
+}
+
+
+/**
+ * fs_stream_transmitter_add_remote_candidates
+ * @streamtransmitter: a #FsStreamTranmitter
+ * @candidates: (element-type FsCandidate): a #GList of the remote candidates
+ * @error: location of a #GError, or NULL if no error occured
+ *
+ * This function is used to add remote candidates to the transmitter
+ *
+ * Returns: TRUE of the candidate could be added, FALSE if it couldnt
+ * (and the #GError will be set)
+ */
+
+gboolean
+fs_stream_transmitter_add_remote_candidates (
+ FsStreamTransmitter *streamtransmitter,
+ GList *candidates,
+ GError **error)
+{
+ FsStreamTransmitterClass *klass;
+
+ g_return_val_if_fail (streamtransmitter, FALSE);
+ g_return_val_if_fail (FS_IS_STREAM_TRANSMITTER (streamtransmitter), FALSE);
+ klass = FS_STREAM_TRANSMITTER_GET_CLASS (streamtransmitter);
+
+ if (klass->add_remote_candidates) {
+ return klass->add_remote_candidates (streamtransmitter, candidates, error);
+ } else {
+ g_set_error (error, FS_ERROR, FS_ERROR_NOT_IMPLEMENTED,
+ "add_remote_candidate not defined in stream transmitter class");
+ }
+
+ return FALSE;
+}
+
+/**
+ * fs_stream_transmitter_force_remote_candidates:
+ * @streamtransmitter: a #FsStreamTransmitter
+ * @remote_candidates: (element-type FsCandidate): a #GList of #FsCandidate to
+ * force
+ * @error: location of a #GError, or NULL if no error occured
+ *
+ * This function forces data to be sent immediately to the selected remote
+ * candidate, by-passing any connectivity checks. There should be at most
+ * one candidate per component.
+ *
+ * Returns: %TRUE if the candidates could be forced, %FALSE otherwise
+ */
+
+gboolean
+fs_stream_transmitter_force_remote_candidates (
+ FsStreamTransmitter *streamtransmitter,
+ GList *remote_candidates,
+ GError **error)
+{
+ FsStreamTransmitterClass *klass;
+
+ g_return_val_if_fail (streamtransmitter, FALSE);
+ g_return_val_if_fail (FS_IS_STREAM_TRANSMITTER (streamtransmitter), FALSE);
+ klass = FS_STREAM_TRANSMITTER_GET_CLASS (streamtransmitter);
+
+ if (klass->force_remote_candidates) {
+ return klass->force_remote_candidates (streamtransmitter,
+ remote_candidates, error);
+ } else {
+ g_set_error (error, FS_ERROR, FS_ERROR_NOT_IMPLEMENTED,
+ "force_remote_candidates not defined in stream transmitter class");
+ }
+
+ return FALSE;
+}
+
+/**
+ * fs_stream_transmitter_gather_local_candidates:
+ * @streamtransmitter: a #FsStreamTransmitter
+ * @error: location of a #GErrorh, or NULL if no error occured
+ *
+ * This function tells the transmitter to start gathering local candidates,
+ * signals for new candidates and newly active candidates can be emitted
+ * during the call to this function.
+ *
+ * Returns: %TRUE if it succeeds (or is not implemented), %FALSE otherwise
+ */
+
+gboolean
+fs_stream_transmitter_gather_local_candidates (
+ FsStreamTransmitter *streamtransmitter,
+ GError **error)
+{
+ FsStreamTransmitterClass *klass;
+
+ g_return_val_if_fail (streamtransmitter, FALSE);
+ g_return_val_if_fail (FS_IS_STREAM_TRANSMITTER (streamtransmitter), FALSE);
+ klass = FS_STREAM_TRANSMITTER_GET_CLASS (streamtransmitter);
+
+ if (klass->gather_local_candidates)
+ return klass->gather_local_candidates (streamtransmitter, error);
+ else
+ return TRUE;
+}
+
+
+
+/**
+ * fs_stream_transmitter_stop:
+ * @streamtransmitter: a #FsStreamTransmitter
+ *
+ * This functions stops the #FsStreamTransmitter, it must be called before
+ * the last reference is dropped.
+ */
+
+void
+fs_stream_transmitter_stop (FsStreamTransmitter *streamtransmitter)
+{
+ FsStreamTransmitterClass *klass;
+
+ g_return_if_fail (streamtransmitter);
+ g_return_if_fail (FS_IS_STREAM_TRANSMITTER (streamtransmitter));
+ klass = FS_STREAM_TRANSMITTER_GET_CLASS (streamtransmitter);
+
+ if (klass->stop)
+ klass->stop (streamtransmitter);
+}
+
+
+/**
+ * fs_stream_transmitter_emit_error:
+ * @streamtransmitter: #FsStreamTransmitter on which to emit the error signal
+ * @error_no: The number of the error
+ * @error_msg: Error message (for the programmer)
+ *
+ * This function emit the "error" signal on a #FsStreamTransmitter, it should
+ * only be called by subclasses.
+ */
+void
+fs_stream_transmitter_emit_error (FsStreamTransmitter *streamtransmitter,
+ gint error_no,
+ const gchar *error_msg)
+{
+ g_signal_emit (streamtransmitter, signals[ERROR_SIGNAL], 0, error_no,
+ error_msg);
+}