Mercurial > pidgin
view libpurple/protocols/jabber/jingle.c @ 23804:cbe97caec684
Use USE_VV instead of USE_FARSIGHT.
author | Mike Ruprecht <maiku@soc.pidgin.im> |
---|---|
date | Wed, 28 May 2008 21:00:40 +0000 |
parents | 16543025aed1 |
children | d048100a43ab |
line wrap: on
line source
/* * purple - Jabber Protocol Plugin * * 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 "config.h" #include "purple.h" #include "jingle.h" #include "xmlnode.h" #include <stdlib.h> #include <string.h> #include <glib.h> #ifdef USE_VV #include <gst/farsight/fs-candidate.h> static gboolean jabber_jingle_session_equal(gconstpointer a, gconstpointer b) { purple_debug_info("jingle", "jabber_jingle_session_equal, comparing %s and %s\n", ((JingleSession *)a)->id, ((JingleSession *)b)->id); return !strcmp(((JingleSession *) a)->id, ((JingleSession *) b)->id); } static JingleSession * jabber_jingle_session_create_internal(JabberStream *js, const char *id) { JingleSession *sess = g_new0(JingleSession, 1); sess->js = js; if (id) { sess->id = g_strdup(id); } else if (js) { /* init the session ID... */ sess->id = jabber_get_next_id(js); } /* insert it into the hash table */ if (!js->sessions) { purple_debug_info("jingle", "Creating hash table for sessions\n"); js->sessions = g_hash_table_new(g_str_hash, g_str_equal); } purple_debug_info("jingle", "inserting session with key: %s into table\n", sess->id); g_hash_table_insert(js->sessions, sess->id, sess); sess->session_started = FALSE; return sess; } JabberStream * jabber_jingle_session_get_js(const JingleSession *sess) { return sess->js; } JingleSession * jabber_jingle_session_create(JabberStream *js) { JingleSession *sess = jabber_jingle_session_create_internal(js, NULL); sess->is_initiator = TRUE; return sess; } JingleSession * jabber_jingle_session_create_by_id(JabberStream *js, const char *id) { JingleSession *sess = jabber_jingle_session_create_internal(js, id); sess->is_initiator = FALSE; return sess; } const char * jabber_jingle_session_get_id(const JingleSession *sess) { return sess->id; } void jabber_jingle_session_destroy(JingleSession *sess) { g_hash_table_remove(sess->js->sessions, sess->id); g_free(sess->id); g_free(sess); } JingleSession * jabber_jingle_session_find_by_id(JabberStream *js, const char *id) { purple_debug_info("jingle", "find_by_id %s\n", id); purple_debug_info("jingle", "hash table: %p\n", js->sessions); purple_debug_info("jingle", "hash table size %d\n", g_hash_table_size(js->sessions)); purple_debug_info("jingle", "lookup: %p\n", g_hash_table_lookup(js->sessions, id)); return (JingleSession *) g_hash_table_lookup(js->sessions, id); } JingleSession *jabber_jingle_session_find_by_jid(JabberStream *js, const char *jid) { GList *values = g_hash_table_get_values(js->sessions); GList *iter = values; for (; iter; iter = iter->next) { JingleSession *session = (JingleSession *)iter->data; if (session->js == js && !strcmp(jid, session->remote_jid)) { g_list_free(values); return session; } } g_list_free(values); return NULL; } GList * jabber_jingle_get_codecs(const xmlnode *description) { GList *codecs = NULL; xmlnode *codec_element = NULL; const char *encoding_name,*id, *clock_rate; FsCodec *codec; for (codec_element = xmlnode_get_child(description, "payload-type") ; codec_element ; codec_element = xmlnode_get_next_twin(codec_element)) { encoding_name = xmlnode_get_attrib(codec_element, "name"); id = xmlnode_get_attrib(codec_element, "id"); clock_rate = xmlnode_get_attrib(codec_element, "clockrate"); codec = fs_codec_new(atoi(id), encoding_name, FS_MEDIA_TYPE_AUDIO, clock_rate ? atoi(clock_rate) : 0); codecs = g_list_append(codecs, codec); } return codecs; } GList * jabber_jingle_get_candidates(const xmlnode *transport) { GList *candidates = NULL; xmlnode *candidate = NULL; FsCandidate *c; for (candidate = xmlnode_get_child(transport, "candidate") ; candidate ; candidate = xmlnode_get_next_twin(candidate)) { const char *type = xmlnode_get_attrib(candidate, "type"); c = fs_candidate_new(xmlnode_get_attrib(candidate, "component"), atoi(xmlnode_get_attrib(candidate, "component")), strcmp(type, "host") == 0 ? FS_CANDIDATE_TYPE_HOST : strcmp(type, "prflx") == 0 ? FS_CANDIDATE_TYPE_PRFLX : FS_CANDIDATE_TYPE_RELAY, strcmp(xmlnode_get_attrib(candidate, "protocol"), "udp") == 0 ? FS_NETWORK_PROTOCOL_UDP : FS_NETWORK_PROTOCOL_TCP, xmlnode_get_attrib(candidate, "ip"), atoi(xmlnode_get_attrib(candidate, "port"))); candidates = g_list_append(candidates, c); } return candidates; } PurpleMedia * jabber_jingle_session_get_media(const JingleSession *sess) { return sess->media; } void jabber_jingle_session_set_media(JingleSession *sess, PurpleMedia *media) { sess->media = media; } const char * jabber_jingle_session_get_remote_jid(const JingleSession *sess) { return sess->remote_jid; } void jabber_jingle_session_set_remote_jid(JingleSession *sess, const char *remote_jid) { sess->remote_jid = strdup(remote_jid); } const char * jabber_jingle_session_get_initiator(const JingleSession *sess) { return sess->initiator; } void jabber_jingle_session_set_initiator(JingleSession *sess, const char *initiator) { sess->initiator = g_strdup(initiator); } gboolean jabber_jingle_session_is_initiator(const JingleSession *sess) { return sess->is_initiator; } static xmlnode * jabber_jingle_session_create_jingle_element(const JingleSession *sess, const char *action) { xmlnode *jingle = xmlnode_new("jingle"); xmlnode_set_namespace(jingle, "urn:xmpp:tmp:jingle"); xmlnode_set_attrib(jingle, "action", action); xmlnode_set_attrib(jingle, "initiator", jabber_jingle_session_get_initiator(sess)); xmlnode_set_attrib(jingle, "responder", jabber_jingle_session_is_initiator(sess) ? jabber_jingle_session_get_remote_jid(sess) : jabber_jingle_session_get_initiator(sess)); xmlnode_set_attrib(jingle, "sid", sess->id); return jingle; } xmlnode * jabber_jingle_session_create_terminate(const JingleSession *sess, const char *reasoncode, const char *reasontext) { xmlnode *jingle = jabber_jingle_session_create_jingle_element(sess, "session-terminate"); xmlnode_set_attrib(jingle, "resoncode", reasoncode); if (reasontext) { xmlnode_set_attrib(jingle, "reasontext", reasontext); } xmlnode_set_attrib(jingle, "sid", sess->id); return jingle; } xmlnode * jabber_jingle_session_create_description(const JingleSession *sess) { GList *codecs = purple_media_get_local_audio_codecs(sess->media); xmlnode *description = xmlnode_new("description"); xmlnode_set_namespace(description, "urn:xmpp:tmp:jingle:apps:audio-rtp"); /* get codecs */ for (; codecs ; codecs = codecs->next) { FsCodec *codec = (FsCodec*)codecs->data; char id[8], clockrate[10], channels[10]; xmlnode *payload = xmlnode_new_child(description, "payload-type"); g_snprintf(id, sizeof(id), "%d", codec->id); g_snprintf(clockrate, sizeof(clockrate), "%d", codec->clock_rate); g_snprintf(channels, sizeof(channels), "%d", codec->channels); xmlnode_set_attrib(payload, "name", codec->encoding_name); xmlnode_set_attrib(payload, "id", id); xmlnode_set_attrib(payload, "clockrate", clockrate); xmlnode_set_attrib(payload, "channels", channels); } fs_codec_list_destroy(codecs); return description; } static guint jabber_jingle_get_priority(guint type, guint network) { switch (type) { case FS_CANDIDATE_TYPE_HOST: return network == 0 ? 4096 : network == 1 ? 2048 : 1024; break; case FS_CANDIDATE_TYPE_PRFLX: return 126; break; case FS_CANDIDATE_TYPE_RELAY: return 100; break; default: return 0; /* unknown type, should not happen */ } } static xmlnode * jabber_jingle_session_create_candidate_info(FsCandidate *c, FsCandidate *remote) { char port[8]; char prio[8]; char component[8]; xmlnode *candidate = NULL; candidate = xmlnode_new("candidate"); g_snprintf(port, sizeof(port), "%d", c->port); g_snprintf(prio, sizeof(prio), "%d", jabber_jingle_get_priority(c->type, 0)); g_snprintf(component, sizeof(component), "%d", c->component_id); xmlnode_set_attrib(candidate, "component", component); xmlnode_set_attrib(candidate, "foundation", "1"); /* what about this? */ xmlnode_set_attrib(candidate, "generation", "0"); /* ? */ xmlnode_set_attrib(candidate, "ip", c->ip); xmlnode_set_attrib(candidate, "network", "0"); /* ? */ xmlnode_set_attrib(candidate, "port", port); xmlnode_set_attrib(candidate, "priority", prio); /* Is this correct? */ xmlnode_set_attrib(candidate, "protocol", c->proto == FS_NETWORK_PROTOCOL_UDP ? "udp" : "tcp"); if (c->username) xmlnode_set_attrib(candidate, "ufrag", c->username); if (c->password) xmlnode_set_attrib(candidate, "pwd", c->password); xmlnode_set_attrib(candidate, "type", c->type == FS_CANDIDATE_TYPE_HOST ? "host" : c->type == FS_CANDIDATE_TYPE_PRFLX ? "prflx" : c->type == FS_CANDIDATE_TYPE_RELAY ? "relay" : NULL); /* relay */ if (c->type == FS_CANDIDATE_TYPE_RELAY) { /* set rel-addr and rel-port? How? */ } if (remote) { char remote_port[8]; g_snprintf(remote_port, sizeof(remote_port), "%d", remote->port); xmlnode_set_attrib(candidate, "rem-addr", remote->ip); xmlnode_set_attrib(candidate, "rem-port", remote_port); } return candidate; } /* split into two separate methods, one to generate session-accept (includes codecs) and one to generate transport-info (includes transports candidates) */ xmlnode * jabber_jingle_session_create_session_accept(const JingleSession *sess) { xmlnode *jingle = jabber_jingle_session_create_jingle_element(sess, "session-accept"); xmlnode *content = NULL; xmlnode *description = NULL; xmlnode *transport = NULL; xmlnode *candidate = NULL; content = xmlnode_new_child(jingle, "content"); xmlnode_set_attrib(content, "creator", "initiator"); xmlnode_set_attrib(content, "name", "audio-content"); xmlnode_set_attrib(content, "profile", "RTP/AVP"); description = jabber_jingle_session_create_description(sess); xmlnode_insert_child(content, description); transport = xmlnode_new_child(content, "transport"); xmlnode_set_namespace(transport, "urn:xmpp:tmp:jingle:transports:ice-udp"); candidate = jabber_jingle_session_create_candidate_info( purple_media_get_local_candidate(sess->media), purple_media_get_remote_candidate(sess->media)); xmlnode_insert_child(transport, candidate); return jingle; } xmlnode * jabber_jingle_session_create_transport_info(const JingleSession *sess) { xmlnode *jingle = jabber_jingle_session_create_jingle_element(sess, "transport-info"); xmlnode *content = NULL; xmlnode *transport = NULL; GList *candidates = purple_media_get_local_audio_candidates(sess->media); content = xmlnode_new_child(jingle, "content"); xmlnode_set_attrib(content, "creator", "initiator"); xmlnode_set_attrib(content, "name", "audio-content"); xmlnode_set_attrib(content, "profile", "RTP/AVP"); transport = xmlnode_new_child(content, "transport"); xmlnode_set_namespace(transport, "urn:xmpp:tmp:jingle:transports:ice-udp"); /* get transport candidate */ for (; candidates ; candidates = candidates->next) { FsCandidate *c = (FsCandidate *) candidates->data; if (!strcmp(c->ip, "127.0.0.1")) { continue; } xmlnode_insert_child(transport, jabber_jingle_session_create_candidate_info(c, NULL)); } fs_candidate_list_destroy(candidates); return jingle; } xmlnode * jabber_jingle_session_create_content_replace(const JingleSession *sess, FsCandidate *native_candidate, FsCandidate *remote_candidate) { xmlnode *jingle = jabber_jingle_session_create_jingle_element(sess, "content-replace"); xmlnode *content = NULL; xmlnode *transport = NULL; purple_debug_info("jingle", "creating content-modify for native candidate %s " \ ", remote candidate %s\n", native_candidate->candidate_id, remote_candidate->candidate_id); content = xmlnode_new_child(jingle, "content"); xmlnode_set_attrib(content, "creator", "initiator"); xmlnode_set_attrib(content, "name", "audio-content"); xmlnode_set_attrib(content, "profile", "RTP/AVP"); /* get top codec from codec_intersection to put here... */ /* later on this should probably handle changing codec */ xmlnode_insert_child(content, jabber_jingle_session_create_description(sess)); transport = xmlnode_new_child(content, "transport"); xmlnode_set_namespace(transport, "urn:xmpp:tmp:jingle:transports:ice-udp"); xmlnode_insert_child(transport, jabber_jingle_session_create_candidate_info(native_candidate, remote_candidate)); purple_debug_info("jingle", "End create content modify\n"); return jingle; } xmlnode * jabber_jingle_session_create_content_accept(const JingleSession *sess) { xmlnode *jingle = jabber_jingle_session_create_jingle_element(sess, "content-accept"); xmlnode *content = xmlnode_new_child(jingle, "content"); xmlnode *description = jabber_jingle_session_create_description(sess); xmlnode_set_attrib(content, "creator", "initiator"); xmlnode_set_attrib(content, "name", "audio-content"); xmlnode_set_attrib(content, "profile", "RTP/AVP"); xmlnode_insert_child(content, description); return jingle; } #endif /* USE_VV */