Mercurial > pidgin
view libpurple/protocols/jabber/jingle/iceudp.c @ 29329:c38f6e6bf4ce
propagate from branch 'im.pidgin.pidgin' (head f0addfa93983499a661e29f567e66a98a9544a89)
to branch 'im.pidgin.pidgin.next.minor' (head 9c3b44f9ecfcb6fde88c8bcc3a27a16587422f2e)
author | Paul Aurich <paul@darkrain42.org> |
---|---|
date | Wed, 09 Dec 2009 03:52:00 +0000 |
parents | c585572e80dd |
children | a8cc50c2279f |
line wrap: on
line source
/** * @file iceudp.c * * purple * * Purple is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * source distribution. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA */ #include "internal.h" #include "iceudp.h" #include "jingle.h" #include "debug.h" #include <string.h> struct _JingleIceUdpPrivate { GList *local_candidates; GList *remote_candidates; }; #define JINGLE_ICEUDP_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE((obj), JINGLE_TYPE_ICEUDP, JingleIceUdpPrivate)) static void jingle_iceudp_class_init (JingleIceUdpClass *klass); static void jingle_iceudp_init (JingleIceUdp *iceudp); static void jingle_iceudp_finalize (GObject *object); static void jingle_iceudp_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void jingle_iceudp_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static JingleTransport *jingle_iceudp_parse_internal(xmlnode *iceudp); static xmlnode *jingle_iceudp_to_xml_internal(JingleTransport *transport, xmlnode *content, JingleActionType action); static JingleTransportClass *parent_class = NULL; enum { PROP_0, PROP_LOCAL_CANDIDATES, PROP_REMOTE_CANDIDATES, }; static JingleIceUdpCandidate * jingle_iceudp_candidate_copy(JingleIceUdpCandidate *candidate) { JingleIceUdpCandidate *new_candidate = g_new0(JingleIceUdpCandidate, 1); new_candidate->component = candidate->component; new_candidate->foundation = g_strdup(candidate->foundation); new_candidate->generation = candidate->generation; new_candidate->id = g_strdup(candidate->id); new_candidate->ip = g_strdup(candidate->ip); new_candidate->network = candidate->network; new_candidate->port = candidate->port; new_candidate->priority = candidate->priority; new_candidate->protocol = g_strdup(candidate->protocol); new_candidate->type = g_strdup(candidate->type); new_candidate->username = g_strdup(candidate->username); new_candidate->password = g_strdup(candidate->password); new_candidate->rem_known = candidate->rem_known; return new_candidate; } static void jingle_iceudp_candidate_free(JingleIceUdpCandidate *candidate) { g_free(candidate->foundation); g_free(candidate->id); g_free(candidate->ip); g_free(candidate->protocol); g_free(candidate->reladdr); g_free(candidate->type); g_free(candidate->username); g_free(candidate->password); } GType jingle_iceudp_candidate_get_type() { static GType type = 0; if (type == 0) { type = g_boxed_type_register_static("JingleIceUdpCandidate", (GBoxedCopyFunc)jingle_iceudp_candidate_copy, (GBoxedFreeFunc)jingle_iceudp_candidate_free); } return type; } JingleIceUdpCandidate * jingle_iceudp_candidate_new(guint component, const gchar *foundation, guint generation, const gchar *id, const gchar *ip, guint network, guint port, guint priority, const gchar *protocol, const gchar *type, const gchar *username, const gchar *password) { JingleIceUdpCandidate *candidate = g_new0(JingleIceUdpCandidate, 1); candidate->component = component; candidate->foundation = g_strdup(foundation); candidate->generation = generation; candidate->id = g_strdup(id); candidate->ip = g_strdup(ip); candidate->network = network; candidate->port = port; candidate->priority = priority; candidate->protocol = g_strdup(protocol); candidate->type = g_strdup(type); candidate->username = g_strdup(username); candidate->password = g_strdup(password); candidate->rem_known = FALSE; return candidate; } GType jingle_iceudp_get_type() { static GType type = 0; if (type == 0) { static const GTypeInfo info = { sizeof(JingleIceUdpClass), NULL, NULL, (GClassInitFunc) jingle_iceudp_class_init, NULL, NULL, sizeof(JingleIceUdp), 0, (GInstanceInitFunc) jingle_iceudp_init, NULL }; type = g_type_register_static(JINGLE_TYPE_TRANSPORT, "JingleIceUdp", &info, 0); } return type; } static void jingle_iceudp_class_init (JingleIceUdpClass *klass) { GObjectClass *gobject_class = (GObjectClass*)klass; parent_class = g_type_class_peek_parent(klass); gobject_class->finalize = jingle_iceudp_finalize; gobject_class->set_property = jingle_iceudp_set_property; gobject_class->get_property = jingle_iceudp_get_property; klass->parent_class.to_xml = jingle_iceudp_to_xml_internal; klass->parent_class.parse = jingle_iceudp_parse_internal; klass->parent_class.transport_type = JINGLE_TRANSPORT_ICEUDP; g_object_class_install_property(gobject_class, PROP_LOCAL_CANDIDATES, g_param_spec_pointer("local-candidates", "Local candidates", "The local candidates for this transport.", G_PARAM_READABLE)); g_object_class_install_property(gobject_class, PROP_REMOTE_CANDIDATES, g_param_spec_pointer("remote-candidates", "Remote candidates", "The remote candidates for this transport.", G_PARAM_READABLE)); g_type_class_add_private(klass, sizeof(JingleIceUdpPrivate)); } static void jingle_iceudp_init (JingleIceUdp *iceudp) { iceudp->priv = JINGLE_ICEUDP_GET_PRIVATE(iceudp); iceudp->priv->local_candidates = NULL; iceudp->priv->remote_candidates = NULL; } static void jingle_iceudp_finalize (GObject *iceudp) { /* JingleIceUdpPrivate *priv = JINGLE_ICEUDP_GET_PRIVATE(iceudp); */ purple_debug_info("jingle","jingle_iceudp_finalize\n"); G_OBJECT_CLASS(parent_class)->finalize(iceudp); } static void jingle_iceudp_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { JingleIceUdp *iceudp; g_return_if_fail(JINGLE_IS_ICEUDP(object)); iceudp = JINGLE_ICEUDP(object); switch (prop_id) { case PROP_LOCAL_CANDIDATES: iceudp->priv->local_candidates = g_value_get_pointer(value); break; case PROP_REMOTE_CANDIDATES: iceudp->priv->remote_candidates = g_value_get_pointer(value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void jingle_iceudp_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { JingleIceUdp *iceudp; g_return_if_fail(JINGLE_IS_ICEUDP(object)); iceudp = JINGLE_ICEUDP(object); switch (prop_id) { case PROP_LOCAL_CANDIDATES: g_value_set_pointer(value, iceudp->priv->local_candidates); break; case PROP_REMOTE_CANDIDATES: g_value_set_pointer(value, iceudp->priv->remote_candidates); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } void jingle_iceudp_add_local_candidate(JingleIceUdp *iceudp, JingleIceUdpCandidate *candidate) { GList *iter = iceudp->priv->local_candidates; for (; iter; iter = g_list_next(iter)) { JingleIceUdpCandidate *c = iter->data; if (!strcmp(c->id, candidate->id)) { guint generation = c->generation + 1; g_boxed_free(JINGLE_TYPE_ICEUDP_CANDIDATE, c); iceudp->priv->local_candidates = g_list_delete_link( iceudp->priv->local_candidates, iter); candidate->generation = generation; iceudp->priv->local_candidates = g_list_append( iceudp->priv->local_candidates, candidate); return; } } iceudp->priv->local_candidates = g_list_append( iceudp->priv->local_candidates, candidate); } GList * jingle_iceudp_get_remote_candidates(JingleIceUdp *iceudp) { return g_list_copy(iceudp->priv->remote_candidates); } static JingleIceUdpCandidate * jingle_iceudp_get_remote_candidate_by_id(JingleIceUdp *iceudp, const gchar *id) { GList *iter = iceudp->priv->remote_candidates; for (; iter; iter = g_list_next(iter)) { JingleIceUdpCandidate *candidate = iter->data; if (!strcmp(candidate->id, id)) { return candidate; } } return NULL; } static void jingle_iceudp_add_remote_candidate(JingleIceUdp *iceudp, JingleIceUdpCandidate *candidate) { JingleIceUdpPrivate *priv = JINGLE_ICEUDP_GET_PRIVATE(iceudp); JingleIceUdpCandidate *iceudp_candidate = jingle_iceudp_get_remote_candidate_by_id(iceudp, candidate->id); if (iceudp_candidate != NULL) { priv->remote_candidates = g_list_remove( priv->remote_candidates, iceudp_candidate); g_boxed_free(JINGLE_TYPE_ICEUDP_CANDIDATE, iceudp_candidate); } priv->remote_candidates = g_list_append(priv->remote_candidates, candidate); } static JingleTransport * jingle_iceudp_parse_internal(xmlnode *iceudp) { JingleTransport *transport = parent_class->parse(iceudp); xmlnode *candidate = xmlnode_get_child(iceudp, "candidate"); JingleIceUdpCandidate *iceudp_candidate = NULL; const gchar *username = xmlnode_get_attrib(iceudp, "ufrag"); const gchar *password = xmlnode_get_attrib(iceudp, "pwd"); for (; candidate; candidate = xmlnode_get_next_twin(candidate)) { const gchar *relport = xmlnode_get_attrib(candidate, "rel-port"); const gchar *component = xmlnode_get_attrib(candidate, "component"); const gchar *foundation = xmlnode_get_attrib(candidate, "foundation"); const gchar *generation = xmlnode_get_attrib(candidate, "generation"); const gchar *id = xmlnode_get_attrib(candidate, "id"); const gchar *ip = xmlnode_get_attrib(candidate, "ip"); const gchar *network = xmlnode_get_attrib(candidate, "network"); const gchar *port = xmlnode_get_attrib(candidate, "port"); const gchar *priority = xmlnode_get_attrib(candidate, "priority"); const gchar *protocol = xmlnode_get_attrib(candidate, "protocol"); const gchar *type = xmlnode_get_attrib(candidate, "type"); if (!component || !foundation || !generation || !id || !ip || !network || !port || !priority || !protocol || !type) continue; iceudp_candidate = jingle_iceudp_candidate_new( atoi(component), foundation, atoi(generation), id, ip, atoi(network), atoi(port), atoi(priority), protocol, type, username, password); iceudp_candidate->reladdr = g_strdup( xmlnode_get_attrib(candidate, "rel-addr")); iceudp_candidate->relport = relport != NULL ? atoi(relport) : 0; iceudp_candidate->rem_known = TRUE; jingle_iceudp_add_remote_candidate(JINGLE_ICEUDP(transport), iceudp_candidate); } return transport; } static xmlnode * jingle_iceudp_to_xml_internal(JingleTransport *transport, xmlnode *content, JingleActionType action) { xmlnode *node = parent_class->to_xml(transport, content, action); if (action == JINGLE_SESSION_INITIATE || action == JINGLE_SESSION_ACCEPT || action == JINGLE_TRANSPORT_INFO || action == JINGLE_CONTENT_ADD || action == JINGLE_TRANSPORT_REPLACE) { JingleIceUdpPrivate *priv = JINGLE_ICEUDP_GET_PRIVATE(transport); GList *iter = priv->local_candidates; gboolean used_candidate = FALSE; for (; iter; iter = g_list_next(iter)) { JingleIceUdpCandidate *candidate = iter->data; xmlnode *xmltransport; gchar *component, *generation, *network, *port, *priority; if (candidate->rem_known == TRUE) continue; used_candidate = TRUE; candidate->rem_known = TRUE; xmltransport = xmlnode_new_child(node, "candidate"); component = g_strdup_printf("%d", candidate->component); generation = g_strdup_printf("%d", candidate->generation); network = g_strdup_printf("%d", candidate->network); port = g_strdup_printf("%d", candidate->port); priority = g_strdup_printf("%d", candidate->priority); xmlnode_set_attrib(xmltransport, "component", component); xmlnode_set_attrib(xmltransport, "foundation", candidate->foundation); xmlnode_set_attrib(xmltransport, "generation", generation); xmlnode_set_attrib(xmltransport, "id", candidate->id); xmlnode_set_attrib(xmltransport, "ip", candidate->ip); xmlnode_set_attrib(xmltransport, "network", network); xmlnode_set_attrib(xmltransport, "port", port); xmlnode_set_attrib(xmltransport, "priority", priority); xmlnode_set_attrib(xmltransport, "protocol", candidate->protocol); if (candidate->reladdr != NULL && (strcmp(candidate->ip, candidate->reladdr) || (candidate->port != candidate->relport))) { gchar *relport = g_strdup_printf("%d", candidate->relport); xmlnode_set_attrib(xmltransport, "rel-addr", candidate->reladdr); xmlnode_set_attrib(xmltransport, "rel-port", relport); g_free(relport); } xmlnode_set_attrib(xmltransport, "type", candidate->type); g_free(component); g_free(generation); g_free(network); g_free(port); g_free(priority); } if (used_candidate == TRUE) { JingleIceUdpCandidate *candidate = priv->local_candidates->data; xmlnode_set_attrib(node, "pwd", candidate->password); xmlnode_set_attrib(node, "ufrag", candidate->username); } } return node; }