/*************************************************************************** * linphone_tunnel.cc * * Fri Dec 9, 2011 * Copyright 2011 Belledonne Communications * Author: Guillaume Beraudo * Email: guillaume dot beraudo at linphone dot org ****************************************************************************/ /* * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "TunnelManager.hh" #include "linphone/tunnel.h" #include "linphone/core.h" #include "private.h" #include "linphone/lpconfig.h" LinphoneTunnel* linphone_core_get_tunnel(const LinphoneCore *lc){ return lc->tunnel; } struct _LinphoneTunnel { ::belle_sip_object_t base; belledonnecomm::TunnelManager *manager; bctbx_list_t *config_list; }; static void _linphone_tunnel_uninit(LinphoneTunnel *tunnel); BELLE_SIP_DECLARE_NO_IMPLEMENTED_INTERFACES(LinphoneTunnel); BELLE_SIP_DECLARE_VPTR(LinphoneTunnel); BELLE_SIP_INSTANCIATE_VPTR(LinphoneTunnel, belle_sip_object_t, _linphone_tunnel_uninit, // uninit NULL, // clone NULL, // marshal FALSE // unowned ) extern "C" LinphoneTunnel* linphone_core_tunnel_new(LinphoneCore *lc){ LinphoneTunnel* tunnel = belle_sip_object_new(LinphoneTunnel); tunnel->manager = new belledonnecomm::TunnelManager(lc); return tunnel; } static void _linphone_tunnel_uninit(LinphoneTunnel *tunnel) { delete tunnel->manager; bctbx_list_free_with_data(tunnel->config_list, (bctbx_list_free_func)linphone_tunnel_config_unref); } belledonnecomm::TunnelManager *bcTunnel(const LinphoneTunnel *tunnel){ return tunnel->manager; } static inline _LpConfig *config(const LinphoneTunnel *tunnel){ return tunnel->manager->getLinphoneCore()->config; } LinphoneTunnel *linphone_tunnel_ref(LinphoneTunnel *tunnel) { return (LinphoneTunnel *)belle_sip_object_ref(tunnel); } void linphone_tunnel_unref(LinphoneTunnel *tunnel) { belle_sip_object_unref(tunnel); } static char *linphone_tunnel_config_to_string(const LinphoneTunnelConfig *tunnel_config) { char *str = NULL; const char *host = linphone_tunnel_config_get_host(tunnel_config); const char *host2 = linphone_tunnel_config_get_host2(tunnel_config); if(host != NULL) { if(linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config) != -1) { if (host2 != NULL) { str = ms_strdup_printf("%s:%d:%d:%d/%s:%d", linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config), linphone_tunnel_config_get_delay(tunnel_config), linphone_tunnel_config_get_host2(tunnel_config), linphone_tunnel_config_get_port2(tunnel_config)); } else { str = ms_strdup_printf("%s:%d:%d:%d", linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config), linphone_tunnel_config_get_delay(tunnel_config)); } } else if (host2 != NULL) { str = ms_strdup_printf("%s:%d/%s:%d", linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_host2(tunnel_config), linphone_tunnel_config_get_port2(tunnel_config)); } else { str = ms_strdup_printf("%s:%d", linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config)); } } return str; } static LinphoneTunnelConfig *linphone_tunnel_config_from_string(const char *str) { LinphoneTunnelConfig *tunnel_config = NULL; char * dstr = ms_strdup(str); const char *host = NULL; int port = -1; int remote_udp_mirror_port = -1; int delay = -1; int pos = 0; char *pch; char *tok1, *tok2; tok1 = strtok(dstr, "/"); tok2 = strtok(NULL, "/"); pch = strtok(tok1, ":"); while (pch != NULL) { switch(pos) { case 0: host = pch; break; case 1: port = atoi(pch); break; case 2: remote_udp_mirror_port = atoi(pch); break; case 3: delay = atoi(pch); break; default: // Abort pos = 0; break; } ++pos; pch = strtok(NULL, ":"); } if (pos >= 2) { tunnel_config = linphone_tunnel_config_new(); linphone_tunnel_config_set_remote_udp_mirror_port(tunnel_config, -1); linphone_tunnel_config_set_host(tunnel_config, host); linphone_tunnel_config_set_port(tunnel_config, port); } if (pos >= 3) { linphone_tunnel_config_set_remote_udp_mirror_port(tunnel_config, remote_udp_mirror_port); } if (pos == 4) { linphone_tunnel_config_set_delay(tunnel_config, delay); } if (tok2) { pos = 0; pch = strtok(tok2, ":"); while (pch != NULL) { switch(pos) { case 0: host = pch; break; case 1: port = atoi(pch); break; case 2: remote_udp_mirror_port = atoi(pch); break; case 3: delay = atoi(pch); break; default: // Abort pos = 0; break; } ++pos; pch = strtok(NULL, ":"); } if (pos >= 2 && tunnel_config) { linphone_tunnel_config_set_host2(tunnel_config, host); linphone_tunnel_config_set_port2(tunnel_config, port); } } ms_free(dstr); return tunnel_config; } static void linphone_tunnel_save_config(const LinphoneTunnel *tunnel) { bctbx_list_t *elem = NULL; char *tmp = NULL, *old_tmp = NULL, *tc_str = NULL; for(elem = tunnel->config_list; elem != NULL; elem = elem->next) { LinphoneTunnelConfig *tunnel_config = (LinphoneTunnelConfig *)elem->data; tc_str = linphone_tunnel_config_to_string(tunnel_config); if(tc_str != NULL) { if(tmp != NULL) { old_tmp = tmp; tmp = ms_strdup_printf("%s %s", old_tmp, tc_str); ms_free(old_tmp); ms_free(tc_str); } else { tmp = tc_str; } } } lp_config_set_string(config(tunnel), "tunnel", "server_addresses", tmp); if(tmp != NULL) { ms_free(tmp); } } static void linphone_tunnel_add_server_intern(LinphoneTunnel *tunnel, LinphoneTunnelConfig *tunnel_config) { if (linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config) != -1) { if (linphone_tunnel_config_get_host2(tunnel_config) != NULL) { bcTunnel(tunnel)->addServerPair(linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_host2(tunnel_config), linphone_tunnel_config_get_port2(tunnel_config), linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config), linphone_tunnel_config_get_delay(tunnel_config)); } else { bcTunnel(tunnel)->addServer(linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_remote_udp_mirror_port(tunnel_config), linphone_tunnel_config_get_delay(tunnel_config)); } } else if (linphone_tunnel_config_get_host2(tunnel_config) != NULL) { bcTunnel(tunnel)->addServerPair(linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config), linphone_tunnel_config_get_host2(tunnel_config), linphone_tunnel_config_get_port2(tunnel_config)); } else { bcTunnel(tunnel)->addServer(linphone_tunnel_config_get_host(tunnel_config), linphone_tunnel_config_get_port(tunnel_config)); } tunnel->config_list = bctbx_list_append(tunnel->config_list, linphone_tunnel_config_ref(tunnel_config)); } static void linphone_tunnel_load_config(LinphoneTunnel *tunnel){ const char * confaddress = lp_config_get_string(config(tunnel), "tunnel", "server_addresses", NULL); char *tmp; const char *it; LinphoneTunnelConfig *tunnel_config; int adv; if(confaddress != NULL) { tmp = ms_strdup(confaddress); it = confaddress; while(confaddress[0] != '\0') { int ret = sscanf(it,"%s%n", tmp, &adv); if (ret >= 1){ it += adv; tunnel_config = linphone_tunnel_config_from_string(tmp); if(tunnel_config != NULL) { linphone_tunnel_add_server_intern(tunnel, tunnel_config); } else { ms_error("Tunnel server address incorrectly specified from config file: %s", tmp); } } else break; } ms_free(tmp); } } static void linphone_tunnel_refresh_config(LinphoneTunnel *tunnel) { bctbx_list_t *old_list = tunnel->config_list; tunnel->config_list = NULL; bcTunnel(tunnel)->cleanServers(); while(old_list != NULL) { LinphoneTunnelConfig *tunnel_config = (LinphoneTunnelConfig *)old_list->data; linphone_tunnel_add_server_intern(tunnel, tunnel_config); linphone_tunnel_config_unref(tunnel_config); old_list = old_list->next; } } void linphone_tunnel_add_server(LinphoneTunnel *tunnel, LinphoneTunnelConfig *tunnel_config) { linphone_tunnel_add_server_intern(tunnel, tunnel_config); linphone_tunnel_save_config(tunnel); } void linphone_tunnel_remove_server(LinphoneTunnel *tunnel, LinphoneTunnelConfig *tunnel_config) { bctbx_list_t *elem = bctbx_list_find(tunnel->config_list, tunnel_config); if(elem != NULL) { tunnel->config_list = bctbx_list_remove(tunnel->config_list, tunnel_config); linphone_tunnel_config_unref(tunnel_config); linphone_tunnel_refresh_config(tunnel); linphone_tunnel_save_config(tunnel); } } const bctbx_list_t *linphone_tunnel_get_servers(const LinphoneTunnel *tunnel){ return tunnel->config_list; } void linphone_tunnel_clean_servers(LinphoneTunnel *tunnel){ bcTunnel(tunnel)->cleanServers(); /* Free the list */ bctbx_list_free_with_data(tunnel->config_list, (void (*)(void *))linphone_tunnel_config_destroy); tunnel->config_list = NULL; linphone_tunnel_save_config(tunnel); } void linphone_tunnel_set_mode(LinphoneTunnel *tunnel, LinphoneTunnelMode mode){ lp_config_set_string(config(tunnel),"tunnel","mode", linphone_tunnel_mode_to_string(mode)); bcTunnel(tunnel)->setMode(mode); } LinphoneTunnelMode linphone_tunnel_get_mode(const LinphoneTunnel *tunnel){ return bcTunnel(tunnel)->getMode(); } void linphone_tunnel_enable_dual_mode(LinphoneTunnel *tunnel, bool_t dual_mode_enabled) { lp_config_set_int(config(tunnel), "tunnel", "dual_mode", (dual_mode_enabled ? TRUE : FALSE)); bcTunnel(tunnel)->enableDualMode(dual_mode_enabled); } bool_t linphone_tunnel_dual_mode_enabled(const LinphoneTunnel *tunnel) { return bcTunnel(tunnel)->isDualModeEnabled(); } bool_t linphone_tunnel_connected(const LinphoneTunnel *tunnel){ return bcTunnel(tunnel)->isConnected(); } bool_t linphone_tunnel_get_activated(const LinphoneTunnel *tunnel){ return bcTunnel(tunnel)->isActivated(); } static OrtpLogFunc tunnelOrtpLogHandler=NULL; /* #define TUNNEL_DEBUG (1) #define TUNNEL_INFO (1<<1) #define TUNNEL_NOTICE (1<<2) #define TUNNEL_WARN (1<<3) #define TUNNEL_ERROR (1<<4) #define TUNNEL_ALERT (1<<5) #define TUNNEL_FATAL (1<<6) */ static void tunnelLogHandler(int level, const char *fmt, va_list l){ if (tunnelOrtpLogHandler){ OrtpLogLevel ortp_level=ORTP_DEBUG; switch(level){ case TUNNEL_DEBUG: ortp_level=ORTP_DEBUG; break; case TUNNEL_INFO: ortp_level=ORTP_MESSAGE; break; case TUNNEL_NOTICE: ortp_level=ORTP_MESSAGE; break; case TUNNEL_WARN: ortp_level=ORTP_WARNING; break; case TUNNEL_ERROR: ortp_level=ORTP_ERROR; break; case TUNNEL_ALERT: ortp_level=ORTP_ERROR; break; case TUNNEL_FATAL: ortp_level=ORTP_FATAL; break; default: ms_fatal("Unexepcted tunnel log %i: %s",level,fmt); break; } tunnelOrtpLogHandler("tunnel", ortp_level,fmt,l); } } void linphone_tunnel_enable_logs_with_handler(LinphoneTunnel *tunnel, bool_t enabled, OrtpLogFunc logHandler){ tunnelOrtpLogHandler=logHandler; bcTunnel(tunnel)->enableLogs(enabled == FALSE ? false : true, tunnelLogHandler); } void linphone_tunnel_set_http_proxy_auth_info(LinphoneTunnel *tunnel, const char* username,const char* passwd){ bcTunnel(tunnel)->setHttpProxyAuthInfo(username, passwd); } void linphone_tunnel_set_http_proxy(LinphoneTunnel*tunnel, const char *host, int port, const char* username,const char* passwd){ bcTunnel(tunnel)->setHttpProxy(host, port, username, passwd); lp_config_set_string(config(tunnel),"tunnel","http_proxy_host",host); lp_config_set_int(config(tunnel),"tunnel","http_proxy_port",port); lp_config_set_string(config(tunnel),"tunnel","http_proxy_username",username); lp_config_set_string(config(tunnel),"tunnel","http_proxy_password",passwd); } void linphone_tunnel_get_http_proxy(LinphoneTunnel*tunnel,const char **host, int *port, const char **username, const char **passwd){ if (host) *host=lp_config_get_string(config(tunnel),"tunnel","http_proxy_host",NULL); if (port) *port=lp_config_get_int(config(tunnel),"tunnel","http_proxy_port",0); if (username) *username=lp_config_get_string(config(tunnel),"tunnel","http_proxy_username",NULL); if (passwd) *passwd=lp_config_get_string(config(tunnel),"tunnel","http_proxy_password",NULL); } void linphone_tunnel_reconnect(LinphoneTunnel *tunnel){ bcTunnel(tunnel)->reconnect(); } void linphone_tunnel_enable_sip(LinphoneTunnel *tunnel, bool_t enable) { bcTunnel(tunnel)->tunnelizeSipPackets(enable == FALSE ? false : true); lp_config_set_int(config(tunnel), "tunnel", "sip", (enable ? TRUE : FALSE)); } bool_t linphone_tunnel_sip_enabled(const LinphoneTunnel *tunnel) { return bcTunnel(tunnel)->tunnelizeSipPacketsEnabled() ? TRUE : FALSE; } void linphone_tunnel_verify_server_certificate(LinphoneTunnel *tunnel, bool_t enable) { bcTunnel(tunnel)->verifyServerCertificate(enable == FALSE ? false : true); lp_config_set_int(config(tunnel), "tunnel", "verify_cert", (enable ? TRUE : FALSE)); } bool_t linphone_tunnel_verify_server_certificate_enabled(const LinphoneTunnel *tunnel) { return bcTunnel(tunnel)->verifyServerCertificateEnabled() ? TRUE : FALSE; } static void my_ortp_logv(const char *domain, OrtpLogLevel level, const char *fmt, va_list args){ ortp_logv(domain, level,fmt,args); } /** * Startup tunnel using configuration. * Called internally from linphonecore at startup. */ void linphone_tunnel_configure(LinphoneTunnel *tunnel){ LinphoneTunnelMode mode = linphone_tunnel_mode_from_string(lp_config_get_string(config(tunnel), "tunnel", "mode", NULL)); bool_t tunnelizeSIPPackets = (bool_t)lp_config_get_int(config(tunnel), "tunnel", "sip", TRUE); bool_t tunnelVerifyServerCertificate = (bool_t)lp_config_get_int(config(tunnel), "tunnel", "verify_cert", FALSE); bool_t useDualMode = (bool_t)lp_config_get_int(config(tunnel), "tunnel", "dual_mode", FALSE); const char *http_host, *http_username, *http_passwd; int http_port; linphone_tunnel_get_http_proxy(tunnel,&http_host, &http_port, &http_username, &http_passwd); bcTunnel(tunnel)->setHttpProxy(http_host, http_port, http_username, http_passwd); linphone_tunnel_enable_dual_mode(tunnel, useDualMode); linphone_tunnel_enable_logs_with_handler(tunnel,TRUE,my_ortp_logv); linphone_tunnel_load_config(tunnel); linphone_tunnel_enable_sip(tunnel, tunnelizeSIPPackets); linphone_tunnel_verify_server_certificate(tunnel, tunnelVerifyServerCertificate); /*Tunnel is started here if mode equals true*/ linphone_tunnel_set_mode(tunnel, mode); } /* Deprecated functions */ void linphone_tunnel_enable(LinphoneTunnel *tunnel, bool_t enabled) { ms_warning("linphone_tunnel_enable is deprecated - please use linphone_tunnel_set_mode instead."); if(enabled) linphone_tunnel_set_mode(tunnel, LinphoneTunnelModeEnable); else linphone_tunnel_set_mode(tunnel, LinphoneTunnelModeDisable); } bool_t linphone_tunnel_enabled(const LinphoneTunnel *tunnel) { return linphone_tunnel_get_mode(tunnel) == LinphoneTunnelModeEnable; } void linphone_tunnel_auto_detect(LinphoneTunnel *tunnel) { linphone_tunnel_set_mode(tunnel, LinphoneTunnelModeAuto); } bool_t linphone_tunnel_auto_detect_enabled(LinphoneTunnel *tunnel) { return linphone_tunnel_get_mode(tunnel) == LinphoneTunnelModeAuto; } void linphone_tunnel_simulate_udp_loss(LinphoneTunnel *tunnel, bool_t enabled) { bcTunnel(tunnel)->simulateUdpLoss(enabled == FALSE ? false : true); }