rtpmanager: improve SDES handling

Store SDES internally as a struct to support multiple PRIV values.
Include all values set in SDES struct when sending RTCP SDES.
This commit is contained in:
Pascal Buhler 2009-08-31 18:42:25 +02:00 committed by Wim Taymans
parent 251401aef1
commit c3448f978e
3 changed files with 151 additions and 181 deletions

View file

@ -1669,6 +1669,7 @@ rtp_session_process_sdes (RTPSession * sess, GstRTCPPacket * packet,
guint32 ssrc; guint32 ssrc;
gboolean changed, created; gboolean changed, created;
RTPSource *source; RTPSource *source;
GstStructure *sdes;
ssrc = gst_rtcp_packet_sdes_get_ssrc (packet); ssrc = gst_rtcp_packet_sdes_get_ssrc (packet);
@ -1681,24 +1682,45 @@ rtp_session_process_sdes (RTPSession * sess, GstRTCPPacket * packet,
if (!source) if (!source)
return; return;
sdes = gst_structure_new ("application/x-rtp-source-sdes", NULL);
more_entries = gst_rtcp_packet_sdes_first_entry (packet); more_entries = gst_rtcp_packet_sdes_first_entry (packet);
j = 0; j = 0;
while (more_entries) { while (more_entries) {
GstRTCPSDESType type; GstRTCPSDESType type;
guint8 len; guint8 len;
guint8 *data; guint8 *data;
gchar *name;
gchar *value;
gst_rtcp_packet_sdes_get_entry (packet, &type, &len, &data); gst_rtcp_packet_sdes_get_entry (packet, &type, &len, &data);
GST_DEBUG ("entry %d, type %d, len %d, data %.*s", j, type, len, len, GST_DEBUG ("entry %d, type %d, len %d, data %.*s", j, type, len, len,
data); data);
changed |= rtp_source_set_sdes (source, type, data, len); if (type == GST_RTCP_SDES_PRIV) {
name = g_strndup ((const gchar *) &data[1], data[0]);
len -= data[0] + 1;
data += data[0] + 1;
} else {
name = g_strdup (gst_rtcp_sdes_type_to_name (type));
}
value = g_strndup ((const gchar *) data, len);
gst_structure_set (sdes, name, G_TYPE_STRING, value, NULL);
g_free (name);
g_free (value);
more_entries = gst_rtcp_packet_sdes_next_entry (packet); more_entries = gst_rtcp_packet_sdes_next_entry (packet);
j++; j++;
} }
changed = rtp_source_set_sdes_struct (source, sdes);
gst_structure_free (sdes);
source->validated = TRUE; source->validated = TRUE;
if (created) if (created)
@ -2296,27 +2318,60 @@ static void
session_sdes (RTPSession * sess, ReportData * data) session_sdes (RTPSession * sess, ReportData * data)
{ {
GstRTCPPacket *packet = &data->packet; GstRTCPPacket *packet = &data->packet;
guint8 *sdes_data; GstStructure *sdes;
guint sdes_len; gint i, n_fields;
/* add SDES packet */ /* add SDES packet */
gst_rtcp_buffer_add_packet (data->rtcp, GST_RTCP_TYPE_SDES, packet); gst_rtcp_buffer_add_packet (data->rtcp, GST_RTCP_TYPE_SDES, packet);
gst_rtcp_packet_sdes_add_item (packet, sess->source->ssrc); gst_rtcp_packet_sdes_add_item (packet, sess->source->ssrc);
rtp_source_get_sdes (sess->source, GST_RTCP_SDES_CNAME, &sdes_data, sdes = rtp_source_get_sdes_struct (sess->source);
&sdes_len);
gst_rtcp_packet_sdes_add_entry (packet, GST_RTCP_SDES_CNAME, sdes_len,
sdes_data);
/* other SDES items must only be added at regular intervals and only when the /* add all fields in the structure, the order is not important. */
* user requests to since it might be a privacy problem */ n_fields = gst_structure_n_fields (sdes);
#if 0 for (i = 0; i < n_fields; ++i) {
gst_rtcp_packet_sdes_add_entry (&packet, GST_RTCP_SDES_NAME, const gchar *field;
strlen (sess->name), (guint8 *) sess->name); const gchar *value;
gst_rtcp_packet_sdes_add_entry (&packet, GST_RTCP_SDES_TOOL, GstRTCPSDESType type;
strlen (sess->tool), (guint8 *) sess->tool);
#endif field = gst_structure_nth_field_name (sdes, i);
if (field == NULL)
continue;
value = gst_structure_get_string (sdes, field);
if (value == NULL)
continue;
type = gst_rtcp_sdes_name_to_type (field);
if (type > GST_RTCP_SDES_END && type < GST_RTCP_SDES_PRIV) {
gst_rtcp_packet_sdes_add_entry (packet, type, strlen (value),
(const guint8 *) value);
} else if (type == GST_RTCP_SDES_PRIV) {
gsize prefix_len;
gsize value_len;
gsize data_len;
guint8 data[256];
/* don't accept entries that are too big */
prefix_len = strlen (field);
if (prefix_len > 255)
continue;
value_len = strlen (value);
if (value_len > 255)
continue;
data_len = 1 + prefix_len + value_len;
if (data_len > 255)
continue;
data[0] = prefix_len;
memcpy (&data[1], field, prefix_len);
memcpy (&data[1 + prefix_len], value, value_len);
gst_rtcp_packet_sdes_add_entry (packet, type, data_len, data);
}
}
gst_structure_free (sdes);
data->has_sdes = TRUE; data->has_sdes = TRUE;
} }

View file

@ -98,16 +98,19 @@ rtp_source_class_init (RTPSourceClass * klass)
/** /**
* RTPSource::sdes * RTPSource::sdes
* *
* The current SDES items of the source. Returns a structure with the * The current SDES items of the source. Returns a structure with name
* following fields: * application/x-rtp-source-sdes and may contain the following fields:
* *
* 'cname' G_TYPE_STRING : The canonical name * 'cname' G_TYPE_STRING : The canonical name
* 'name' G_TYPE_STRING : The user name * 'name' G_TYPE_STRING : The user name
* 'email' G_TYPE_STRING : The user's electronic mail address * 'email' G_TYPE_STRING : The user's electronic mail address
* 'phone' G_TYPE_STRING : The user's phone number * 'phone' G_TYPE_STRING : The user's phone number
* 'location' G_TYPE_STRING : The geographic user location * 'location' G_TYPE_STRING : The geographic user location
* 'tool' G_TYPE_STRING : The name of application or tool * 'tool' G_TYPE_STRING : The name of application or tool
* 'note' G_TYPE_STRING : A notice about the source * 'note' G_TYPE_STRING : A notice about the source
*
* other fields may be present and these represent private items in
* the SDES where the field name is the prefix.
*/ */
g_object_class_install_property (gobject_class, PROP_SDES, g_object_class_install_property (gobject_class, PROP_SDES,
g_param_spec_boxed ("sdes", "SDES", g_param_spec_boxed ("sdes", "SDES",
@ -156,6 +159,8 @@ rtp_source_init (RTPSource * src)
src->internal = FALSE; src->internal = FALSE;
src->probation = RTP_DEFAULT_PROBATION; src->probation = RTP_DEFAULT_PROBATION;
src->sdes = gst_structure_new ("application/x-rtp-source-sdes", NULL);
src->payload = -1; src->payload = -1;
src->clock_rate = -1; src->clock_rate = -1;
src->packets = g_queue_new (); src->packets = g_queue_new ();
@ -170,7 +175,6 @@ rtp_source_finalize (GObject * object)
{ {
RTPSource *src; RTPSource *src;
GstBuffer *buffer; GstBuffer *buffer;
gint i;
src = RTP_SOURCE_CAST (object); src = RTP_SOURCE_CAST (object);
@ -178,8 +182,7 @@ rtp_source_finalize (GObject * object)
gst_buffer_unref (buffer); gst_buffer_unref (buffer);
g_queue_free (src->packets); g_queue_free (src->packets);
for (i = 0; i < 9; i++) gst_structure_free (src->sdes);
g_free (src->sdes[i]);
g_free (src->bye_reason); g_free (src->bye_reason);
@ -281,88 +284,64 @@ rtp_source_create_stats (RTPSource * src)
/** /**
* rtp_source_get_sdes_struct: * rtp_source_get_sdes_struct:
* @src: an #RTSPSource * @src: an #RTPSource
* *
* Get the SDES data as a GstStructure * Get the SDES from @src.
* *
* Returns: a GstStructure with SDES items for @src. * Returns: %GstStructure of type "application/x-rtp-source-sdes", see
* the SDES property for more details.
*/ */
GstStructure * GstStructure *
rtp_source_get_sdes_struct (RTPSource * src) rtp_source_get_sdes_struct (RTPSource * src)
{ {
GstStructure *s; g_return_val_if_fail (RTP_IS_SOURCE (src), NULL);
gchar *str;
s = gst_structure_new ("application/x-rtp-source-sdes", return gst_structure_copy (src->sdes);
"ssrc", G_TYPE_UINT, (guint) src->ssrc, NULL); }
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_CNAME))) { static gboolean
gst_structure_set (s, "cname", G_TYPE_STRING, str, NULL); sdes_struct_compare_func (GQuark field_id, const GValue * value,
g_free (str); gpointer user_data)
} {
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_NAME))) { GstStructure *old = GST_STRUCTURE (user_data);
gst_structure_set (s, "name", G_TYPE_STRING, str, NULL); const gchar *field = g_quark_to_string (field_id);
g_free (str);
} if (!gst_structure_has_field (old, field))
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_EMAIL))) { return FALSE;
gst_structure_set (s, "email", G_TYPE_STRING, str, NULL);
g_free (str); g_assert (G_VALUE_HOLDS_STRING (value));
} return strcmp (g_value_get_string (value), gst_structure_get_string (old,
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_PHONE))) { field)) == 0;
gst_structure_set (s, "phone", G_TYPE_STRING, str, NULL);
g_free (str);
}
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_LOC))) {
gst_structure_set (s, "location", G_TYPE_STRING, str, NULL);
g_free (str);
}
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_TOOL))) {
gst_structure_set (s, "tool", G_TYPE_STRING, str, NULL);
g_free (str);
}
if ((str = rtp_source_get_sdes_string (src, GST_RTCP_SDES_NOTE))) {
gst_structure_set (s, "note", G_TYPE_STRING, str, NULL);
g_free (str);
}
return s;
} }
/** /**
* rtp_source_set_sdes_struct: * rtp_source_set_sdes:
* @src: an #RTSPSource * @src: an #RTPSource
* @sdes: a #GstStructure with SDES info * @sdes: the SDES structure
* *
* Set the SDES items from @sdes. * Store the @sdes in @src. @sdes must be a structure of type
* "application/x-rtp-source-sdes", see the SDES property for more details.
*
* Returns: %FALSE if the SDES was unchanged.
*/ */
void gboolean
rtp_source_set_sdes_struct (RTPSource * src, const GstStructure * sdes) rtp_source_set_sdes_struct (RTPSource * src, const GstStructure * sdes)
{ {
const gchar *str; gboolean changed;
if (!gst_structure_has_name (sdes, "application/x-rtp-source-sdes")) g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
return;
if ((str = gst_structure_get_string (sdes, "cname"))) { g_return_val_if_fail (strcmp (gst_structure_get_name (sdes),
rtp_source_set_sdes_string (src, GST_RTCP_SDES_CNAME, str); "application/x-rtp-source-sdes") == 0, FALSE);
}
if ((str = gst_structure_get_string (sdes, "name"))) { changed = !gst_structure_foreach (sdes, sdes_struct_compare_func, src->sdes);
rtp_source_set_sdes_string (src, GST_RTCP_SDES_NAME, str);
} if (changed) {
if ((str = gst_structure_get_string (sdes, "email"))) { gst_structure_free (src->sdes);
rtp_source_set_sdes_string (src, GST_RTCP_SDES_EMAIL, str); src->sdes = gst_structure_copy (sdes);
}
if ((str = gst_structure_get_string (sdes, "phone"))) {
rtp_source_set_sdes_string (src, GST_RTCP_SDES_PHONE, str);
}
if ((str = gst_structure_get_string (sdes, "location"))) {
rtp_source_set_sdes_string (src, GST_RTCP_SDES_LOC, str);
}
if ((str = gst_structure_get_string (sdes, "tool"))) {
rtp_source_set_sdes_string (src, GST_RTCP_SDES_TOOL, str);
}
if ((str = gst_structure_get_string (sdes, "note"))) {
rtp_source_set_sdes_string (src, GST_RTCP_SDES_NOTE, str);
} }
return changed;
} }
static void static void
@ -656,55 +635,13 @@ rtp_source_update_caps (RTPSource * src, GstCaps * caps)
gst_caps_replace (&src->caps, caps); gst_caps_replace (&src->caps, caps);
} }
/**
* rtp_source_set_sdes:
* @src: an #RTPSource
* @type: the type of the SDES item
* @data: the SDES data
* @len: the SDES length
*
* Store an SDES item of @type in @src.
*
* Returns: %FALSE if the SDES item was unchanged or @type is unknown.
*/
gboolean
rtp_source_set_sdes (RTPSource * src, GstRTCPSDESType type,
const guint8 * data, guint len)
{
guint8 *old;
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
if (type < 0 || type > GST_RTCP_SDES_PRIV)
return FALSE;
old = src->sdes[type];
/* lengths are the same, check if the data is the same */
if ((src->sdes_len[type] == len))
if (data != NULL && old != NULL && (memcmp (old, data, len) == 0))
return FALSE;
/* NULL data, make sure we store 0 length or if no length is given,
* take strlen */
if (data == NULL)
len = 0;
g_free (src->sdes[type]);
src->sdes[type] = g_memdup (data, len);
src->sdes_len[type] = len;
return TRUE;
}
/** /**
* rtp_source_set_sdes_string: * rtp_source_set_sdes_string:
* @src: an #RTPSource * @src: an #RTPSource
* @type: the type of the SDES item * @type: the type of the SDES item
* @data: the SDES data * @data: the SDES data
* *
* Store an SDES item of @type in @src. This function is similar to * Store an SDES item of @type in @src.
* rtp_source_set_sdes() but takes a null-terminated string for convenience.
* *
* Returns: %FALSE if the SDES item was unchanged or @type is unknown. * Returns: %FALSE if the SDES item was unchanged or @type is unknown.
*/ */
@ -712,48 +649,26 @@ gboolean
rtp_source_set_sdes_string (RTPSource * src, GstRTCPSDESType type, rtp_source_set_sdes_string (RTPSource * src, GstRTCPSDESType type,
const gchar * data) const gchar * data)
{ {
guint len; const gchar *old;
gboolean result; const gchar *field;
if (data) field = gst_rtcp_sdes_type_to_name (type);
len = strlen (data);
if (gst_structure_has_field (src->sdes, field))
old = gst_structure_get_string (src->sdes, field);
else else
len = 0; old = NULL;
result = rtp_source_set_sdes (src, type, (guint8 *) data, len); if (old == NULL && data == NULL)
return result;
}
/**
* rtp_source_get_sdes:
* @src: an #RTPSource
* @type: the type of the SDES item
* @data: location to store the SDES data or NULL
* @len: location to store the SDES length or NULL
*
* Get the SDES item of @type from @src. Note that @data does not always point
* to a null-terminated string, use rtp_source_get_sdes_string() to retrieve a
* null-terminated string instead.
*
* @data remains valid until the next call to rtp_source_set_sdes().
*
* Returns: %TRUE if @type was valid and @data and @len contain valid
* data. @data can be NULL when the item was unset.
*/
gboolean
rtp_source_get_sdes (RTPSource * src, GstRTCPSDESType type, guint8 ** data,
guint * len)
{
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
if (type < 0 || type > GST_RTCP_SDES_PRIV)
return FALSE; return FALSE;
if (data) if (old != NULL && data != NULL && strcmp (old, data) == 0)
*data = src->sdes[type]; return FALSE;
if (len)
*len = src->sdes_len[type]; if (data == NULL)
gst_structure_remove_field (src->sdes, field);
else
gst_structure_set (src->sdes, field, G_TYPE_STRING, data, NULL);
return TRUE; return TRUE;
} }
@ -772,13 +687,19 @@ gchar *
rtp_source_get_sdes_string (RTPSource * src, GstRTCPSDESType type) rtp_source_get_sdes_string (RTPSource * src, GstRTCPSDESType type)
{ {
gchar *result; gchar *result;
const gchar *type_name;
g_return_val_if_fail (RTP_IS_SOURCE (src), NULL); g_return_val_if_fail (RTP_IS_SOURCE (src), NULL);
if (type < 0 || type > GST_RTCP_SDES_PRIV) if (type < 0 || type > GST_RTCP_SDES_PRIV - 1)
return NULL; return NULL;
result = g_strndup ((const gchar *) src->sdes[type], src->sdes_len[type]); type_name = gst_rtcp_sdes_type_to_name (type);
if (!gst_structure_has_field (src->sdes, type_name))
return NULL;
result = g_strdup (gst_structure_get_string (src->sdes, type_name));
return result; return result;
} }

View file

@ -117,8 +117,7 @@ struct _RTPSource {
gboolean is_csrc; gboolean is_csrc;
gboolean is_sender; gboolean is_sender;
guint8 *sdes[9]; GstStructure *sdes;
guint sdes_len[9];
gboolean received_bye; gboolean received_bye;
gchar *bye_reason; gchar *bye_reason;
@ -179,16 +178,11 @@ gchar * rtp_source_get_bye_reason (RTPSource *src);
void rtp_source_update_caps (RTPSource *src, GstCaps *caps); void rtp_source_update_caps (RTPSource *src, GstCaps *caps);
/* SDES info */ /* SDES info */
gboolean rtp_source_set_sdes (RTPSource *src, GstRTCPSDESType type,
const guint8 *data, guint len);
gboolean rtp_source_set_sdes_string (RTPSource *src, GstRTCPSDESType type, gboolean rtp_source_set_sdes_string (RTPSource *src, GstRTCPSDESType type,
const gchar *data); const gchar *data);
gboolean rtp_source_get_sdes (RTPSource *src, GstRTCPSDESType type,
guint8 **data, guint *len);
gchar* rtp_source_get_sdes_string (RTPSource *src, GstRTCPSDESType type); gchar* rtp_source_get_sdes_string (RTPSource *src, GstRTCPSDESType type);
GstStructure * rtp_source_get_sdes_struct (RTPSource * src); GstStructure * rtp_source_get_sdes_struct (RTPSource * src);
void rtp_source_set_sdes_struct (RTPSource * src, const GstStructure *sdes); gboolean rtp_source_set_sdes_struct (RTPSource * src, const GstStructure *sdes);
/* handling network address */ /* handling network address */
void rtp_source_set_rtp_from (RTPSource *src, GstNetAddress *address); void rtp_source_set_rtp_from (RTPSource *src, GstNetAddress *address);