/* eXosip - This is the eXtended osip library. Copyright (C) 2001-2020 Aymeric MOIZARD amoizard@antisip.com eXosip 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. eXosip 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 In addition, as a special exception, the copyright holders give permission to link the code of portions of this program with the OpenSSL library under certain conditions as described in each individual source file, and distribute linked combinations including the two. You must obey the GNU General Public License in all respects for all of the code used other than OpenSSL. If you modify file(s) with this exception, you may extend this exception to your version of the file(s), but you are not obligated to do so. If you do not wish to do so, delete this exception statement from your version. If you delete this exception statement from all source files in the program, then also delete it here. */ #include "eXosip2.h" #include /* Private functions */ static void rcvregister_failure(osip_transaction_t *tr, osip_message_t *sip); static void _eXosip_register_contact_is_modified(eXosip_reg_t *jr, osip_message_t *request, osip_message_t *response) { osip_via_t *via; osip_generic_param_t *br = NULL; char *received_viahost = NULL; char *rport_viaport = NULL; static char *port_5061 = "5061"; static char *port_5060 = "5060"; osip_contact_t *contact; char *contact_port = NULL; /* no action if expires=0 */ if (jr->r_reg_period == 0) return; osip_message_get_via(response, 0, &via); if (via == NULL || via->protocol == NULL || via->host == NULL) return; osip_message_get_contact(request, 0, &contact); if (contact == NULL || contact->url == NULL || contact->url->host == NULL) return; /* via host:port info */ osip_via_param_get_byname(via, "received", &br); if (br != NULL && br->gvalue != NULL) received_viahost = br->gvalue; osip_via_param_get_byname(via, "rport", &br); if (br != NULL && br->gvalue == NULL) return; /* the feature doesn't work if no rport is provided back */ if (br != NULL && br->gvalue != NULL) rport_viaport = br->gvalue; if (rport_viaport == NULL) rport_viaport = via->port; if (rport_viaport == NULL) { /* could be 5060 or 5061 */ if (osip_strcasecmp(via->protocol, "DTLS") == 0 || osip_strcasecmp(via->protocol, "TLS") == 0) rport_viaport = port_5061; else rport_viaport = port_5060; } /* contact host:port info */ contact_port = contact->url->port; if (contact_port == NULL) { /* could be 5060 or 5061 */ if (osip_strcasecmp(via->protocol, "DTLS") == 0 || osip_strcasecmp(via->protocol, "TLS") == 0) contact_port = port_5061; else contact_port = port_5060; } if (jr->r_last_deletion > 0) return; /* avoid loop. */ if (osip_strcasecmp(contact_port, rport_viaport) != 0) { /* the port parameter is different from contact. */ jr->registration_step = RS_DELETIONREQUIRED; /* proceed with deletion and re-registration of new contact */ jr->r_last_deletion = osip_getsystemtime(NULL); return; } if (received_viahost != NULL && osip_strcasecmp(received_viahost, contact->url->host) != 0) { /* a received parameter was added and is different from contact. */ jr->registration_step = RS_DELETIONREQUIRED; /* proceed with deletion and re-registration of new contact */ jr->r_last_deletion = osip_getsystemtime(NULL); return; } if (received_viahost == NULL) { if (osip_strcasecmp(via->host, contact->url->host) != 0) { /* a received parameter was not added and local ip is different from contact: there is a possibility of IP change */ jr->registration_step = RS_DELETIONREQUIRED; /* proceed with deletion and re-registration of new contact */ jr->r_last_deletion = osip_getsystemtime(NULL); } return; } } int _eXosip_snd_message(struct eXosip_t *excontext, osip_transaction_t *tr, osip_message_t *sip, char *host, int port, int out_socket) { int i; osip_via_t *via; if (sip->status_code == 101) return OSIP_SUCCESS; via = (osip_via_t *) osip_list_get(&sip->vias, 0); if (via == NULL || via->protocol == NULL) return -1; if (host == NULL) { if (MSG_IS_REQUEST(sip)) { osip_route_t *route; osip_message_get_route(sip, 0, &route); if (route != NULL) { osip_uri_param_t *lr_param = NULL; osip_uri_uparam_get_byname(route->url, "lr", &lr_param); if (lr_param == NULL) route = NULL; } if (route != NULL) { port = 5060; if (route->url->port != NULL) port = osip_atoi(route->url->port); host = route->url->host; } else { /* search for maddr parameter */ osip_uri_param_t *maddr_param = NULL; osip_uri_param_t *obr_param = NULL; osip_uri_param_t *obp_param = NULL; osip_uri_uparam_get_byname(sip->req_uri, "maddr", &maddr_param); host = NULL; if (maddr_param != NULL && maddr_param->gvalue != NULL) host = maddr_param->gvalue; port = 5060; if (sip->req_uri->port != NULL) port = osip_atoi(sip->req_uri->port); if (host == NULL) { /* if ob was used in Contact, then exosip adds "x-obr" and "x-obp", thus, when processing request, the ip/port destination are re-used here */ osip_uri_uparam_get_byname(sip->req_uri, "x-obr", &obr_param); osip_uri_uparam_get_byname(sip->req_uri, "x-obp", &obp_param); if (obr_param != NULL && obr_param->gvalue != NULL && obp_param != NULL && obp_param->gvalue != NULL) { host = obr_param->gvalue; port = atoi(obp_param->gvalue); } } if (host == NULL) host = sip->req_uri->host; } } else { osip_generic_param_t *maddr; osip_generic_param_t *received; osip_generic_param_t *rport; osip_via_param_get_byname(via, "maddr", &maddr); osip_via_param_get_byname(via, "received", &received); osip_via_param_get_byname(via, "rport", &rport); if (maddr != NULL && maddr->gvalue != NULL) host = maddr->gvalue; else if (received != NULL && received->gvalue != NULL) host = received->gvalue; else host = via->host; if (rport == NULL || rport->gvalue == NULL) { if (via->port != NULL) port = osip_atoi(via->port); else port = 5060; } else port = osip_atoi(rport->gvalue); } } if (excontext->cbsipCallback != NULL) { excontext->cbsipCallback(sip, 0); } i = -1; i = excontext->eXtl_transport.tl_send_message(excontext, tr, sip, host, port, out_socket); if (i != 0) { return i; } return OSIP_SUCCESS; } static int cb_snd_message(osip_transaction_t *tr, osip_message_t *sip, char *host, int port, int out_socket) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); return _eXosip_snd_message(excontext, tr, sip, host, port, out_socket); } static void cb_xixt_kill_transaction(int type, osip_transaction_t *tr) { int i; struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); struct timeval time_sub; osip_timersub(&tr->destroyed_time, &tr->created_time, &time_sub); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_xixt_kill_transaction] [duration=%3li,%03lis]\n", tr->transactionid, time_sub.tv_sec, time_sub.tv_usec / 1000)); i = osip_remove_transaction(excontext->j_osip, tr); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_BUG, NULL, "[eXosip] [tid=%i] [cb_xixt_kill_transaction] **cannot remove transaction from the oSIP stack**\n", tr->transactionid)); } if (MSG_IS_REGISTER(tr->orig_request) && type == OSIP_NICT_KILL_TRANSACTION && tr->last_response == NULL) { rcvregister_failure(tr, NULL); return; } #ifndef MINISIZE if (type == OSIP_NICT_KILL_TRANSACTION) { eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); if (jc == NULL && jn == NULL && js == NULL && tr->last_response == NULL) { eXosip_event_t *je; je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_REQUESTFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (jc != NULL && jn == NULL && js == NULL && tr->last_response == NULL) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_REQUESTFAILURE, jc, jd, tr); return; } /* no answer to a NOTIFY request! */ if (jn != NULL && MSG_IS_NOTIFY(tr->orig_request) && tr->last_response == NULL) { /* delete the dialog! */ eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_REQUESTFAILURE, jn, jd, tr); _eXosip_report_event(excontext, je); REMOVE_ELEMENT(excontext->j_notifies, jn); _eXosip_notify_free(excontext, jn); return; } if (jn != NULL && MSG_IS_NOTIFY(tr->orig_request) && tr->last_response != NULL && tr->last_response->status_code > 299) { /* delete the dialog! */ if (tr->last_response->status_code != 407 && tr->last_response->status_code != 401) { REMOVE_ELEMENT(excontext->j_notifies, jn); _eXosip_notify_free(excontext, jn); return; } } if (jn != NULL && MSG_IS_NOTIFY(tr->orig_request) && tr->last_response != NULL && tr->last_response->status_code > 199 && tr->last_response->status_code < 300) { if (jn->n_ss_status == EXOSIP_SUBCRSTATE_TERMINATED) { /* delete the dialog! */ REMOVE_ELEMENT(excontext->j_notifies, jn); _eXosip_notify_free(excontext, jn); return; } } /* no answer to a SUBSCRIBE request! */ if (js != NULL && (MSG_IS_SUBSCRIBE(tr->orig_request) || MSG_IS_REFER(tr->orig_request)) && (tr->last_response == NULL || tr->last_response->status_code <= 199)) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_REQUESTFAILURE, js, jd, tr); _eXosip_report_event(excontext, je); /* delete the dialog! */ REMOVE_ELEMENT(excontext->j_subscribes, js); _eXosip_subscription_free(excontext, js); return; } /* detect SUBSCRIBE request that close the dialogs! */ if (js != NULL && (MSG_IS_SUBSCRIBE(tr->orig_request) || MSG_IS_REFER(tr->orig_request)) && tr->last_response != NULL && (tr->last_response->status_code == 401 || tr->last_response->status_code == 407)) { /* delete the dialog later because we need to authenticate */ } else if (js != NULL && (MSG_IS_SUBSCRIBE(tr->orig_request) || MSG_IS_REFER(tr->orig_request))) { osip_header_t *expires; osip_message_get_expires(tr->orig_request, 0, &expires); if (expires == NULL || expires->hvalue == NULL) { } else if (0 == strcmp(expires->hvalue, "0")) { REMOVE_ELEMENT(excontext->j_subscribes, js); _eXosip_subscription_free(excontext, js); return; } } } #endif } static void cb_rcvcancel(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvcancel]\n", tr->transactionid)); if ((jd != NULL) && (jc != NULL)) { /* do propagate CANCEL to the application */ _eXosip_report_call_event(excontext, EXOSIP_CALL_CANCELLED, jc, jd, tr); } } static void cb_rcvregister(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_event_t *je; OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvregister]\n", tr->transactionid)); je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_NEW, tr); _eXosip_event_add(excontext, je); return; } static void cb_rcvrequest(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); #ifndef MINISIZE eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); #endif OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvrequest]\n", tr->transactionid)); if (jc != NULL) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvrequest]\n", tr->transactionid)); if (MSG_IS_BYE(sip)) { if (excontext->autoanswer_bye == 0) { /* not already sent */ _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_NEW, jc, jd, tr); _eXosip_report_call_event(excontext, EXOSIP_CALL_CLOSED, jc, jd, tr); } } else _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_NEW, jc, jd, tr); return; } #ifndef MINISIZE else if (jn != NULL) { if (MSG_IS_SUBSCRIBE(sip) || MSG_IS_REFER(sip)) { eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_IN_SUBSCRIPTION_NEW, jn, jd, tr); _eXosip_report_event(excontext, je); return; } return; } else if (js != NULL) { if (MSG_IS_NOTIFY(sip)) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_NOTIFY, js, jd, tr); _eXosip_report_event(excontext, je); return; } return; } #endif else { eXosip_event_t *je; je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_NEW, tr); _eXosip_event_add(excontext, je); return; } } void _eXosip_delete_reserved(osip_transaction_t *transaction) { if (transaction == NULL) return; osip_transaction_set_reserved2(transaction, NULL); osip_transaction_set_reserved3(transaction, NULL); osip_transaction_set_reserved4(transaction, NULL); osip_transaction_set_reserved5(transaction, NULL); } static void cb_rcv1xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); #ifndef MINISIZE eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); #endif OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv1xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "OPTIONS")) { if (jc == NULL) { eXosip_event_t *je; OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv1xx] OPTIONS outside of any call\n", tr->transactionid)); je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_PROCEEDING, tr); _eXosip_event_add(excontext, je); return; } _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_PROCEEDING, jc, jd, tr); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE") && MSG_TEST_CODE(sip, 100)) { _eXosip_report_call_event(excontext, EXOSIP_CALL_PROCEEDING, jc, jd, tr); return; } #ifndef MINISIZE if (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") && MSG_TEST_CODE(sip, 100)) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_PROCEEDING, js, jd, tr); _eXosip_report_event(excontext, je); return; } #endif if (MSG_IS_RESPONSE_FOR(sip, "REFER") && MSG_TEST_CODE(sip, 100)) { /* not easy to differentiate now (EXOSIP_SUBSCRIPTION_PROCEEDING or EXOSIP_CALL_MESSAGE_PROCEEDING) */ return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE") #ifndef MINISIZE || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") #endif ) { int i; #ifndef MINISIZE /* for SUBSCRIBE, test if the dialog has been already created with a previous NOTIFY */ if (jd == NULL && js != NULL && js->s_dialogs != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { /* find if existing dialog match the to tag */ osip_generic_param_t *tag; int i; i = osip_to_get_tag(sip->to, &tag); if (i == 0 && tag != NULL && tag->gvalue != NULL) { for (jd = js->s_dialogs; jd != NULL; jd = jd->next) { if (0 == strcmp(jd->d_dialog->remote_tag, tag->gvalue)) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_rcv1xx] found established early dialog for this subscription\n", tr->transactionid)); osip_transaction_set_reserved3(tr, jd); break; } } } } #endif if (jd == NULL) { /* This transaction initiate a dialog in the case of INVITE (else it would be attached to a "jd" element. */ /* allocate a jd */ i = _eXosip_dialog_init_as_uac(&jd, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv1xx] cannot establish a dialog\n", tr->transactionid)); return; } if (jc != NULL) { ADD_ELEMENT(jc->c_dialogs, jd); osip_transaction_set_reserved3(tr, jd); _eXosip_update(excontext); } #ifndef MINISIZE else if (js != NULL) { ADD_ELEMENT(js->s_dialogs, jd); osip_transaction_set_reserved3(tr, jd); _eXosip_update(excontext); } else if (jn != NULL) { ADD_ELEMENT(jn->n_dialogs, jd); osip_transaction_set_reserved3(tr, jd); _eXosip_update(excontext); } #endif else { } } else { if (jd->d_dialog == NULL) { } else if (jd->d_dialog->remote_tag == NULL) { osip_dialog_update_route_set_as_uac(jd->d_dialog, sip); osip_dialog_update_tag_as_uac(jd->d_dialog, sip); } else { osip_generic_param_t *tag; int i; i = osip_to_get_tag(sip->to, &tag); if (tag != NULL && tag->gvalue != NULL && 0 == strcmp(jd->d_dialog->remote_tag, tag->gvalue)) { /* Update only if it is the same dialog */ osip_dialog_update_route_set_as_uac(jd->d_dialog, sip); } #if 1 /* to be tested */ else { /* the best thing is to replace the current dialog information... Much easier than creating a useless dialog! */ int current_local_cseq = jd->d_dialog->local_cseq; osip_dialog_free(jd->d_dialog); i = osip_dialog_init_as_uac(&(jd->d_dialog), sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv1xx] cannot replace the dialog\n", tr->transactionid)); } else { jd->d_dialog->local_cseq = current_local_cseq; OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_WARNING, NULL, "[eXosip] [tid=%i] [cb_rcv1xx] the dialog has been replaced with the new one from 1xx\n", tr->transactionid)); } } #endif } } if (jd != NULL && MSG_IS_RESPONSE_FOR(sip, "INVITE") && sip->status_code < 180) { _eXosip_check_allow_header(jd, sip); _eXosip_report_call_event(excontext, EXOSIP_CALL_PROCEEDING, jc, jd, tr); } else if (jd != NULL && MSG_IS_RESPONSE_FOR(sip, "INVITE") && sip->status_code >= 180) { _eXosip_check_allow_header(jd, sip); _eXosip_report_call_event(excontext, EXOSIP_CALL_RINGING, jc, jd, tr); } #ifndef MINISIZE else if (jd != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_PROCEEDING, js, jd, tr); _eXosip_report_event(excontext, je); } #endif if (jc != NULL && MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_call_renew_expire_time(jc); } } } static void cb_rcv2xx_4invite(osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); int i; eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); if (jc == NULL) return; if (jd == NULL) { /* This transaction initiate a dialog in the case of INVITE (else it would be attached to a "jd" element. */ /* allocate a jd */ i = _eXosip_dialog_init_as_uac(&jd, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv2xx_4invite] cannot establish a dialog\n", tr->transactionid)); return; } ADD_ELEMENT(jc->c_dialogs, jd); osip_transaction_set_reserved3(tr, jd); _eXosip_update(excontext); } else { /* Here is a special case: We have initiated a dialog and we have received informationnal answers from 2 or more remote SIP UA. Those answer can be differentiated with the "To" header's tag. We have used the first informationnal answer to create a dialog, but we now want to be sure the 200ok received is for the dialog this dialog. We have to check the To tag and if it does not match, we just have to modify the existing dialog and replace it. */ osip_generic_param_t *tag; int i; i = osip_to_get_tag(sip->to, &tag); i = 1; /* default is the same dialog */ if (jd->d_dialog == NULL) { /* There are real use-case where a BYE is received/processed before the 200ok of the previous INVITE. In this case, jd->d_dialog is empty and the transaction should be silently discarded. */ /* a ACK should still be sent... -but there is no dialog built- */ return; } if (jd->d_dialog->remote_tag == NULL && tag == NULL) { } /* non compliant remote UA -> assume it is the same dialog */ else if (jd->d_dialog->remote_tag != NULL && tag == NULL) { i = 0; } /* different dialog! */ else if (jd->d_dialog->remote_tag == NULL && tag != NULL) { i = 0; } /* different dialog! */ else if (jd->d_dialog->remote_tag != NULL && tag != NULL && tag->gvalue != NULL && 0 != strcmp(jd->d_dialog->remote_tag, tag->gvalue)) { i = 0; } /* different dialog! */ if (i == 1) { /* just update the dialog */ osip_dialog_update_route_set_as_uac(jd->d_dialog, sip); if (jd->d_dialog->remote_tag == NULL) osip_dialog_update_tag_as_uac(jd->d_dialog, sip); osip_dialog_set_state(jd->d_dialog, DIALOG_CONFIRMED); } else { /* the best thing is to replace the current dialog information... Much easier than creating a useless dialog! */ int current_local_cseq = jd->d_dialog->local_cseq; osip_dialog_free(jd->d_dialog); i = osip_dialog_init_as_uac(&(jd->d_dialog), sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv2xx_4invite] cannot replace the dialog\n", tr->transactionid)); } else { jd->d_dialog->local_cseq = current_local_cseq; OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_WARNING, NULL, "[eXosip] [tid=%i] [cb_rcv2xx_4invite] the dialog has been replaced with the new one from 200ok\n", tr->transactionid)); } } } if (jd != NULL) { osip_header_t *se_exp = NULL; osip_header_t *se_exp_answer = NULL; _eXosip_check_allow_header(jd, sip); osip_message_header_get_byname(tr->orig_request, "session-expires", 0, &se_exp); if (se_exp == NULL) osip_message_header_get_byname(tr->orig_request, "x", 0, &se_exp); osip_message_header_get_byname(sip, "session-expires", 0, &se_exp_answer); if (se_exp_answer == NULL) osip_message_header_get_byname(sip, "x", 0, &se_exp_answer); if (se_exp != NULL && se_exp_answer != NULL) { osip_content_disposition_t *exp_h = NULL; /* syntax of Session-Expires is equivalent to "Content-Disposition" */ osip_content_disposition_init(&exp_h); if (exp_h != NULL) { osip_content_disposition_parse(exp_h, se_exp_answer->hvalue); if (exp_h->element != NULL) { osip_generic_param_t *param = NULL; osip_generic_param_get_byname(&exp_h->gen_params, "refresher", ¶m); if (param == NULL) { jd->d_refresher = 0; /* me? in which case? */ } else { if (osip_strcasecmp(param->gvalue, "uac") == 0) jd->d_refresher = 0; else jd->d_refresher = 1; } jd->d_session_timer_start = osip_getsystemtime(NULL); jd->d_session_timer_length = atoi(exp_h->element); if (jd->d_session_timer_length <= 90) jd->d_session_timer_length = 90; } osip_content_disposition_free(exp_h); exp_h = NULL; } } else if (se_exp != NULL && se_exp_answer == NULL && excontext->opt_sessiontimers_force > 0) { /* not supported on remote end, but we want timer at UAC with */ osip_content_disposition_t *exp_h = NULL; /* syntax of Session-Expires is equivalent to "Content-Disposition" */ osip_content_disposition_init(&exp_h); if (exp_h != NULL) { osip_content_disposition_parse(exp_h, se_exp->hvalue); if (exp_h->element != NULL) { jd->d_refresher = 0; jd->d_session_timer_start = osip_getsystemtime(NULL); jd->d_session_timer_length = atoi(exp_h->element); if (jd->d_session_timer_length <= 90) jd->d_session_timer_length = 90; } osip_content_disposition_free(exp_h); exp_h = NULL; } } } /* _eXosip_dialog_set_200ok (jd, sip); */ _eXosip_report_call_event(excontext, EXOSIP_CALL_ANSWERED, jc, jd, tr); /* look for the SDP information and decide if this answer was for an initial INVITE, an HoldCall, or a RetreiveCall */ /* don't handle hold/unhold by now... */ /* _eXosip_update_audio_session(tr); */ } #ifndef MINISIZE static void cb_rcv2xx_4subscribe(osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); int i; eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); if (MSG_IS_RESPONSE_FOR(sip, "REFER")) { osip_header_t *refer_sub; osip_message_header_get_byname(sip, "Refer-Sub", 0, &refer_sub); if (refer_sub != NULL && refer_sub->hvalue != NULL && osip_strncasecmp(refer_sub->hvalue, "false", 5) == 0) { /* do not create JD, deletion will be immediate */ eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_ANSWERED, js, jd, tr); _eXosip_report_event(excontext, je); return; } } _eXosip_subscription_set_refresh_interval(js, sip); /* for SUBSCRIBE, test if the dialog has been already created with a previous NOTIFY */ if (jd == NULL && js != NULL && js->s_dialogs != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { /* find if existing dialog match the to tag */ osip_generic_param_t *tag; int i; i = osip_to_get_tag(sip->to, &tag); if (i == 0 && tag != NULL && tag->gvalue != NULL) { for (jd = js->s_dialogs; jd != NULL; jd = jd->next) { if (0 == strcmp(jd->d_dialog->remote_tag, tag->gvalue)) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_rcv2xx_4subscribe] found established early dialog for this subscription\n", tr->transactionid)); osip_transaction_set_reserved3(tr, jd); break; } } } } if (jd == NULL) { /* This transaction initiate a dialog in the case of SUBSCRIBE (else it would be attached to a "jd" element. */ /* allocate a jd */ i = _eXosip_dialog_init_as_uac(&jd, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv2xx_4subscribe] cannot establish a dialog\n", tr->transactionid)); return; } ADD_ELEMENT(js->s_dialogs, jd); osip_transaction_set_reserved3(tr, jd); _eXosip_update(excontext); } else { osip_dialog_update_route_set_as_uac(jd->d_dialog, sip); if (jd->d_dialog->remote_tag == NULL) osip_dialog_update_tag_as_uac(jd->d_dialog, sip); osip_dialog_set_state(jd->d_dialog, DIALOG_CONFIRMED); } /* look for the body information */ { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_ANSWERED, js, jd, tr); _eXosip_report_event(excontext, je); } } #endif static int _eXosip_update_expires_according_to_contact(eXosip_reg_t *jreg, osip_transaction_t *tr, osip_message_t *sip) { osip_list_iterator_t it; osip_contact_t *co_register; int maxval = 0; if (jreg == NULL) return OSIP_BADPARAMETER; /* only update if only one Contact was in INVITE */ if (tr->orig_request == NULL) return OSIP_BADPARAMETER; if (osip_list_size(&tr->orig_request->contacts) != 1) return OSIP_SUCCESS; /* search for matching contact (line parameter must be equal) */ co_register = (osip_contact_t *) osip_list_get_first(&sip->contacts, &it); while (co_register != OSIP_SUCCESS) { osip_uri_param_t *line_param = NULL; if (co_register->url != NULL) osip_uri_uparam_get_byname(co_register->url, "line", &line_param); if (line_param != NULL && line_param->gvalue != NULL) { if (osip_strcasecmp(jreg->r_line, line_param->gvalue) == 0) { /* found contact */ int val; osip_generic_param_t *exp_param = NULL; osip_contact_param_get_byname(co_register, "expires", &exp_param); if (exp_param != NULL && exp_param->gvalue != NULL) { val = atoi(exp_param->gvalue); if (val > maxval) { maxval = val; } } } } co_register = (osip_contact_t *) osip_list_get_next(&it); } if (maxval == 0) return OSIP_NOTFOUND; /* update only if expires value has REALLY be decreased (more than one minutes): In many cases value is decreased because a few seconds has elapsed when server send the 200ok. */ if (maxval < jreg->r_reg_period - 15) { jreg->r_reg_period = maxval; } else if (maxval > jreg->r_reg_period) { jreg->r_reg_period = maxval; } return OSIP_SUCCESS; } static void cb_rcv2xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); #ifndef MINISIZE eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); #endif osip_header_t *sub_state; osip_header_t *refer_sub; time_t now = osip_getsystemtime(NULL); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv2xx]\n", tr->transactionid)); #ifndef MINISIZE if (MSG_IS_RESPONSE_FOR(sip, "PUBLISH")) { eXosip_pub_t *pub = NULL; eXosip_event_t *je; int i; i = _eXosip_pub_update(excontext, &pub, tr, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv2xx] no publication to update\n", tr->transactionid)); } if (pub != NULL) { /* update registration interval */ osip_header_t *exp; osip_message_header_get_byname(sip, "expires", 0, &exp); if (exp != NULL && exp->hvalue != NULL) { int val = atoi(exp->hvalue); if (val > 0) { /* update only if expires value has REALLY be decreased (more than one minutes): In many cases value is decreased because a few seconds has elapsed when server send the 200ok. */ if (val < pub->p_period - 15) { pub->p_period = val; } } } pub->p_retry = 0; /* reset value */ } je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_ANSWERED, tr); _eXosip_report_event(excontext, je); return; } #endif if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { eXosip_event_t *je; eXosip_reg_t *jreg = NULL; /* find matching j_reg */ _eXosip_reg_find(excontext, &jreg, tr); if (jreg != NULL) { if (jreg->registration_step != RS_DELETIONPROCEEDING) { /* step to remove contact */ /* update registration interval */ osip_header_t *exp; osip_message_header_get_byname(sip, "expires", 0, &exp); if (exp != NULL && exp->hvalue != NULL) { int val = atoi(exp->hvalue); if (val > 0) { /* update only if expires value has REALLY be decreased (more than one minutes): In many cases value is decreased because a few seconds has elapsed when server send the 200ok. */ if (val < jreg->r_reg_period - 15) { jreg->r_reg_period = val; } } } _eXosip_update_expires_according_to_contact(jreg, tr, sip); } if (jreg->registration_step == RS_DELETIONPROCEEDING) jreg->registration_step = RS_MASQUERADINGREQUIRED; /* final registration with correct contact to be done */ if (jreg->registration_step > RS_MASQUERADINGREQUIRED) _eXosip_register_contact_is_modified(jreg, tr->orig_request, sip); je = _eXosip_event_init_for_reg(EXOSIP_REGISTRATION_SUCCESS, jreg, tr); _eXosip_report_event(excontext, je); jreg->r_retry = 0; /* reset value */ jreg->r_retryfailover = 0; } return; } if (jd != NULL) jd->d_retry = 0; /* reset marker for authentication */ if (jc != NULL) jc->c_retry = 0; /* reset marker for authentication */ #ifndef MINISIZE if (js != NULL) js->s_retry = 0; /* reset marker for authentication */ #endif if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { cb_rcv2xx_4invite(tr, sip); return; } if (jc != NULL) { if (MSG_IS_RESPONSE_FOR(sip, "BYE")) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_ANSWERED, jc, jd, tr); return; } if (jd != NULL) _eXosip_check_allow_header(jd, sip); if (MSG_IS_RESPONSE_FOR(sip, "UPDATE")) { if (jd != NULL) { osip_header_t *se_exp = NULL; osip_header_t *se_exp_answer = NULL; osip_message_header_get_byname(tr->orig_request, "session-expires", 0, &se_exp); if (se_exp == NULL) osip_message_header_get_byname(tr->orig_request, "x", 0, &se_exp); osip_message_header_get_byname(sip, "session-expires", 0, &se_exp_answer); if (se_exp_answer == NULL) osip_message_header_get_byname(sip, "x", 0, &se_exp_answer); if (se_exp != NULL && se_exp_answer != NULL) { osip_content_disposition_t *exp_h = NULL; /* syntax of Session-Expires is equivalent to "Content-Disposition" */ osip_content_disposition_init(&exp_h); if (exp_h != NULL) { osip_content_disposition_parse(exp_h, se_exp_answer->hvalue); if (exp_h->element != NULL) { osip_generic_param_t *param = NULL; osip_generic_param_get_byname(&exp_h->gen_params, "refresher", ¶m); if (param == NULL) { jd->d_refresher = 0; /* me? in which case? */ } else { if (osip_strcasecmp(param->gvalue, "uac") == 0) jd->d_refresher = 0; else jd->d_refresher = 1; } jd->d_session_timer_start = osip_getsystemtime(NULL); jd->d_session_timer_length = atoi(exp_h->element); if (jd->d_session_timer_length <= 90) jd->d_session_timer_length = 90; } osip_content_disposition_free(exp_h); exp_h = NULL; } } else if (se_exp != NULL && se_exp_answer == NULL && excontext->opt_sessiontimers_force > 0) { /* not supported on remote end, but we want timer at UAC with */ osip_content_disposition_t *exp_h = NULL; /* syntax of Session-Expires is equivalent to "Content-Disposition" */ osip_content_disposition_init(&exp_h); if (exp_h != NULL) { osip_content_disposition_parse(exp_h, se_exp->hvalue); if (exp_h->element != NULL) { jd->d_refresher = 0; jd->d_session_timer_start = osip_getsystemtime(NULL); jd->d_session_timer_length = atoi(exp_h->element); if (jd->d_session_timer_length <= 90) jd->d_session_timer_length = 90; } osip_content_disposition_free(exp_h); exp_h = NULL; } } } } else if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY")) { if (jd != NULL) { /* get subscription-state */ jd->implicit_subscription_expire_time = 0; osip_message_header_get_byname(tr->orig_request, "subscription-state", 0, &sub_state); if (sub_state != NULL && sub_state->hvalue != NULL) { if (0 == osip_strncasecmp(sub_state->hvalue, "active", 6) || 0 == osip_strncasecmp(sub_state->hvalue, "pending", 7)) { const char *tmp = strstr(sub_state->hvalue + 6, "expires"); const char *ss_expires = NULL; if (tmp != NULL) { ss_expires = strchr(tmp + 7, '='); jd->implicit_subscription_expire_time = now + excontext->implicit_subscription_expires; if (ss_expires != NULL) { int exp; ss_expires++; exp = osip_atoi(ss_expires); if (exp >= 0 && exp < 600) { jd->implicit_subscription_expire_time = now + exp; } } } OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv2xx] dialog marked for implicit-subscription [did=%i]\n", tr->transactionid, jd->d_id)); } } } } else if (MSG_IS_RESPONSE_FOR(sip, "REFER")) { if (jd != NULL) { /* check for "Refer-Sub: false" */ osip_message_header_get_byname(tr->orig_request, "Refer-Sub", 0, &refer_sub); jd->implicit_subscription_expire_time = now + excontext->implicit_subscription_expires; if ((refer_sub != NULL) && (refer_sub->hvalue != NULL) && (0 == osip_strncasecmp(refer_sub->hvalue, "false", 5))) { /* implicit subscription removed */ jd->implicit_subscription_expire_time = 0; OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_WARNING, NULL, "[eXosip] [tid=%i] [cb_rcv2xx] dialog un-marked for implicit-subscription [did=%i]\n", tr->transactionid, jd->d_id)); } } } _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_ANSWERED, jc, jd, tr); return; } #ifndef MINISIZE if (js != NULL) { if (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER")) { cb_rcv2xx_4subscribe(tr, sip); return; } return; } if (jn != NULL) { if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY") && jn != NULL) { eXosip_event_t *je; osip_header_t *sub_state; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_ANSWERED, jn, jd, tr); _eXosip_report_event(excontext, je); osip_message_header_get_byname(tr->orig_request, "subscription-state", 0, &sub_state); if (sub_state == NULL || sub_state->hvalue == NULL) { /* UNCOMPLIANT UA without a subscription-state header */ } else if (0 == osip_strncasecmp(sub_state->hvalue, "terminated", 10)) { /* delete the dialog! */ if (jn != NULL) { REMOVE_ELEMENT(excontext->j_notifies, jn); _eXosip_notify_free(excontext, jn); } } return; } return; } #endif { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_ANSWERED, tr); _eXosip_report_event(excontext, je); return; } } static void _eXosip_delete_early_dialog(struct eXosip_t *excontext, eXosip_dialog_t *jd) { if (jd == NULL) /* bug? */ return; /* an early dialog was created, but the call is not established */ if (jd->d_dialog != NULL && jd->d_dialog->state == DIALOG_EARLY) { osip_dialog_free(jd->d_dialog); jd->d_dialog = NULL; _eXosip_update(excontext); /* AMD 30/09/05 */ } } static void rcvregister_failure(osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_event_t *je; eXosip_reg_t *jreg = NULL; /* find matching j_reg */ _eXosip_reg_find(excontext, &jreg, tr); if (jreg != NULL) { /* proceed with deletion and re-registration of new contact */ if (jreg->registration_step > RS_MASQUERADINGREQUIRED && sip != NULL) _eXosip_register_contact_is_modified(jreg, tr->orig_request, sip); je = _eXosip_event_init_for_reg(EXOSIP_REGISTRATION_FAILURE, jreg, tr); _eXosip_report_event(excontext, je); if (tr->naptr_record != NULL && sip != NULL && sip->status_code == 503) { /* no matter which one, we'll move them all */ osip_gettimeofday(&tr->naptr_record->sipudp_record.srventry[tr->naptr_record->sipudp_record.index].srv_is_broken, NULL); osip_gettimeofday(&tr->naptr_record->siptcp_record.srventry[tr->naptr_record->siptcp_record.index].srv_is_broken, NULL); osip_gettimeofday(&tr->naptr_record->siptls_record.srventry[tr->naptr_record->siptls_record.index].srv_is_broken, NULL); osip_gettimeofday(&tr->naptr_record->sipdtls_record.srventry[tr->naptr_record->sipdtls_record.index].srv_is_broken, NULL); osip_gettimeofday(&tr->naptr_record->sipsctp_record.srventry[tr->naptr_record->sipsctp_record.index].srv_is_broken, NULL); _eXosip_mark_registration_expired(excontext, sip->call_id->number); } } } #ifndef MINISIZE static void cb_rcv3xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv3xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "PUBLISH")) { eXosip_event_t *je; eXosip_pub_t *pub; int i; i = _eXosip_pub_update(excontext, &pub, tr, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv3xx] no publication to update\n", tr->transactionid)); } je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_REDIRECTED, tr); _eXosip_report_event(excontext, je); return; } if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { rcvregister_failure(tr, sip); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_report_call_event(excontext, EXOSIP_CALL_REDIRECTED, jc, jd, tr); } else if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY") && jn != NULL) { eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_REDIRECTED, jn, jd, tr); _eXosip_report_event(excontext, je); } else if (js != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_REDIRECTED, js, jd, tr); _eXosip_report_event(excontext, je); } else if (jc != NULL) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_REDIRECTED, jc, jd, tr); return; } else if (jc == NULL && js == NULL && jn == NULL) { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_REDIRECTED, tr); _eXosip_report_event(excontext, je); return; } if (jd == NULL) return; if (MSG_IS_RESPONSE_FOR(sip, "INVITE") || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE")) { _eXosip_delete_early_dialog(excontext, jd); } } static void cb_rcv4xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv4xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "PUBLISH")) { eXosip_pub_t *pub; eXosip_event_t *je; int i; i = _eXosip_pub_update(excontext, &pub, tr, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv4xx] no publication to update\n", tr->transactionid)); } /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_REQUESTFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { rcvregister_failure(tr, sip); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_report_call_event(excontext, EXOSIP_CALL_REQUESTFAILURE, jc, jd, tr); } else if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY") && jn != NULL) { eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_REQUESTFAILURE, jn, jd, tr); _eXosip_report_event(excontext, je); } else if (js != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_REQUESTFAILURE, js, jd, tr); _eXosip_report_event(excontext, je); } else if (jc != NULL) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_REQUESTFAILURE, jc, jd, tr); return; } else if (jc == NULL && js == NULL && jn == NULL) { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_REQUESTFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (jd == NULL) return; if (MSG_IS_RESPONSE_FOR(sip, "INVITE") || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE")) { _eXosip_delete_early_dialog(excontext, jd); } } static void cb_rcv5xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv5xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "PUBLISH")) { eXosip_pub_t *pub; eXosip_event_t *je; int i; i = _eXosip_pub_update(excontext, &pub, tr, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv5xx] no publication to update\n", tr->transactionid)); } je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_SERVERFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { rcvregister_failure(tr, sip); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_report_call_event(excontext, EXOSIP_CALL_SERVERFAILURE, jc, jd, tr); } else if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY") && jn != NULL) { eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_SERVERFAILURE, jn, jd, tr); _eXosip_report_event(excontext, je); } else if (js != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_SERVERFAILURE, js, jd, tr); _eXosip_report_event(excontext, je); } else if (jc != NULL) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_SERVERFAILURE, jc, jd, tr); return; } else if (jc == NULL && js == NULL && jn == NULL) { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_SERVERFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (jd == NULL) return; if (MSG_IS_RESPONSE_FOR(sip, "INVITE") || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE")) { _eXosip_delete_early_dialog(excontext, jd); } } static void cb_rcv6xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv6xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "PUBLISH")) { eXosip_pub_t *pub; eXosip_event_t *je; int i; i = _eXosip_pub_update(excontext, &pub, tr, sip); if (i != 0) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_ERROR, NULL, "[eXosip] [tid=%i] [cb_rcv6xx] no publication to update\n", tr->transactionid)); } je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_GLOBALFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { rcvregister_failure(tr, sip); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_report_call_event(excontext, EXOSIP_CALL_GLOBALFAILURE, jc, jd, tr); } else if (MSG_IS_RESPONSE_FOR(sip, "NOTIFY") && jn != NULL) { eXosip_event_t *je; je = _eXosip_event_init_for_notify(EXOSIP_NOTIFICATION_GLOBALFAILURE, jn, jd, tr); _eXosip_report_event(excontext, je); } else if (js != NULL && (MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE") || MSG_IS_RESPONSE_FOR(sip, "REFER"))) { eXosip_event_t *je; je = _eXosip_event_init_for_subscription(EXOSIP_SUBSCRIPTION_GLOBALFAILURE, js, jd, tr); _eXosip_report_event(excontext, je); } else if (jc != NULL) { _eXosip_report_call_event(excontext, EXOSIP_CALL_MESSAGE_GLOBALFAILURE, jc, jd, tr); return; } else if (jc == NULL && js == NULL && jn == NULL) { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(EXOSIP_MESSAGE_GLOBALFAILURE, tr); _eXosip_report_event(excontext, je); return; } if (jd == NULL) return; if (MSG_IS_RESPONSE_FOR(sip, "INVITE") || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE")) { _eXosip_delete_early_dialog(excontext, jd); } } #else static void cb_rcv3456xx(int type, osip_transaction_t *tr, osip_message_t *sip, int invite_event, int call_event, int outcall_event); static void cb_rcv3456xx(int type, osip_transaction_t *tr, osip_message_t *sip, int invite_event, int call_event, int outcall_event) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcv3456xx]\n", tr->transactionid)); if (MSG_IS_RESPONSE_FOR(sip, "REGISTER")) { rcvregister_failure(tr, sip); return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_report_call_event(excontext, invite_event, jc, jd, tr); } else if (jc != NULL) { _eXosip_report_call_event(excontext, call_event, jc, jd, tr); return; } else if (jc == NULL) { eXosip_event_t *je; /* For all requests outside of calls */ je = _eXosip_event_init_for_message(outcall_event, tr); _eXosip_report_event(excontext, je); return; } if (jd == NULL) return; if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { _eXosip_delete_early_dialog(excontext, jd); } } static void cb_rcv3xx(int type, osip_transaction_t *tr, osip_message_t *sip) { cb_rcv3456xx(type, tr, sip, EXOSIP_CALL_REDIRECTED, EXOSIP_CALL_MESSAGE_REDIRECTED, EXOSIP_MESSAGE_REDIRECTED); } static void cb_rcv4xx(int type, osip_transaction_t *tr, osip_message_t *sip) { cb_rcv3456xx(type, tr, sip, EXOSIP_CALL_REQUESTFAILURE, EXOSIP_CALL_MESSAGE_REQUESTFAILURE, EXOSIP_MESSAGE_REQUESTFAILURE); } static void cb_rcv5xx(int type, osip_transaction_t *tr, osip_message_t *sip) { cb_rcv3456xx(type, tr, sip, EXOSIP_CALL_SERVERFAILURE, EXOSIP_CALL_MESSAGE_SERVERFAILURE, EXOSIP_MESSAGE_SERVERFAILURE); } static void cb_rcv6xx(int type, osip_transaction_t *tr, osip_message_t *sip) { cb_rcv3456xx(type, tr, sip, EXOSIP_CALL_GLOBALFAILURE, EXOSIP_CALL_MESSAGE_GLOBALFAILURE, EXOSIP_MESSAGE_GLOBALFAILURE); } #endif static void cb_snd123456xx(int type, osip_transaction_t *tr, osip_message_t *sip) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_snd123456xx]\n", tr->transactionid)); if (jd == NULL) return; if (type == OSIP_IST_STATUS_1XX_SENT || type == OSIP_NIST_STATUS_1XX_SENT) { return; } if (type == OSIP_IST_STATUS_2XX_SENT || type == OSIP_NIST_STATUS_2XX_SENT) { return; } if (MSG_IS_RESPONSE_FOR(sip, "INVITE") || MSG_IS_RESPONSE_FOR(sip, "REFER") || MSG_IS_RESPONSE_FOR(sip, "SUBSCRIBE")) { _eXosip_delete_early_dialog(excontext, jd); } if (MSG_IS_RESPONSE_FOR(sip, "INVITE")) { /* only close calls if this is the initial INVITE */ if (jc != NULL && tr == jc->c_inc_tr) { _eXosip_report_call_event(excontext, EXOSIP_CALL_CLOSED, jc, jd, tr); } } } #ifndef MINISIZE static void cb_rcvinvite(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvinvite]\n", tr->transactionid)); } static void cb_rcvack(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvack]\n", tr->transactionid)); } static void cb_rcvack2(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_rcvack2]\n", tr->transactionid)); } static void cb_sndinvite(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndinvite]\n", tr->transactionid)); } static void cb_sndack(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndack]\n", tr->transactionid)); } static void cb_sndregister(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndregister]\n", tr->transactionid)); } static void cb_sndbye(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndbye]\n", tr->transactionid)); } static void cb_sndcancel(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndcancel]\n", tr->transactionid)); } static void cb_sndinfo(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndinfo]\n", tr->transactionid)); } static void cb_sndoptions(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndoptions]\n", tr->transactionid)); } static void cb_sndnotify(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndnotify]\n", tr->transactionid)); } static void cb_sndsubscribe(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndsubscribe]\n", tr->transactionid)); } static void cb_sndunkrequest(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO3, NULL, "[eXosip] [tid=%i] [cb_sndunkrequest]\n", tr->transactionid)); } static void cb_rcvresp_retransmission(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_rcvresp_retransmission]\n", tr->transactionid)); } static void cb_sndreq_retransmission(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_sndreq_retransmission]\n", tr->transactionid)); } static void cb_sndresp_retransmission(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_sndresp_retransmission]\n", tr->transactionid)); } static void cb_rcvreq_retransmission(int type, osip_transaction_t *tr, osip_message_t *sip) { OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_rcvreq_retransmission]\n", tr->transactionid)); } #endif static void cb_transport_error(int type, osip_transaction_t *tr, int error) { struct eXosip_t *excontext = (struct eXosip_t *) osip_transaction_get_reserved1(tr); #ifndef MINISIZE eXosip_subscribe_t *js = (eXosip_subscribe_t *) osip_transaction_get_reserved5(tr); eXosip_notify_t *jn = (eXosip_notify_t *) osip_transaction_get_reserved4(tr); #endif OSIP_TRACE(osip_trace(__FILE__, __LINE__, OSIP_INFO1, NULL, "[eXosip] [tid=%i] [cb_transport_error]\n", tr->transactionid)); if (type == OSIP_ICT_TRANSPORT_ERROR) { eXosip_call_t *jc = (eXosip_call_t *) osip_transaction_get_reserved2(tr); eXosip_dialog_t *jd = (eXosip_dialog_t *) osip_transaction_get_reserved3(tr); if (jc == NULL && jd == NULL) return; _eXosip_report_call_event(excontext, EXOSIP_CALL_NOANSWER, jc, jd, tr); } if (type == OSIP_NICT_TRANSPORT_ERROR) { osip_naptr_t *naptr_record = tr->naptr_record; if (naptr_record != NULL && naptr_record->naptr_state == OSIP_NAPTR_STATE_SRVDONE && (MSG_IS_REGISTER(tr->orig_request) || MSG_IS_OPTIONS(tr->orig_request))) { if (osip_strcasecmp(excontext->eXtl_transport.proto_name, "UDP") == 0) eXosip_dnsutils_rotate_srv(&naptr_record->sipudp_record); else if (osip_strcasecmp(excontext->eXtl_transport.proto_name, "TCP") == 0) eXosip_dnsutils_rotate_srv(&naptr_record->siptcp_record); else if (osip_strcasecmp(excontext->eXtl_transport.proto_name, "TLS") == 0) eXosip_dnsutils_rotate_srv(&naptr_record->siptls_record); else if (osip_strcasecmp(excontext->eXtl_transport.proto_name, "DTLS-UDP") == 0) eXosip_dnsutils_rotate_srv(&naptr_record->sipdtls_record); } } #ifndef MINISIZE if (jn == NULL && js == NULL) return; if (jn != NULL && MSG_IS_NOTIFY(tr->orig_request) && type == OSIP_NICT_TRANSPORT_ERROR) { /* delete the dialog! */ REMOVE_ELEMENT(excontext->j_notifies, jn); _eXosip_notify_free(excontext, jn); } if (js != NULL && (MSG_IS_SUBSCRIBE(tr->orig_request) || MSG_IS_REFER(tr->orig_request)) && type == OSIP_NICT_TRANSPORT_ERROR) { /* delete the dialog! */ REMOVE_ELEMENT(excontext->j_subscribes, js); _eXosip_subscription_free(excontext, js); } #endif } int _eXosip_set_callbacks(osip_t *osip) { /* register all callbacks */ osip_set_cb_send_message(osip, &cb_snd_message); osip_set_kill_transaction_callback(osip, OSIP_ICT_KILL_TRANSACTION, &cb_xixt_kill_transaction); osip_set_kill_transaction_callback(osip, OSIP_IST_KILL_TRANSACTION, &cb_xixt_kill_transaction); osip_set_kill_transaction_callback(osip, OSIP_NICT_KILL_TRANSACTION, &cb_xixt_kill_transaction); osip_set_kill_transaction_callback(osip, OSIP_NIST_KILL_TRANSACTION, &cb_xixt_kill_transaction); osip_set_message_callback(osip, OSIP_ICT_STATUS_1XX_RECEIVED, &cb_rcv1xx); osip_set_message_callback(osip, OSIP_ICT_STATUS_2XX_RECEIVED, &cb_rcv2xx); osip_set_message_callback(osip, OSIP_ICT_STATUS_3XX_RECEIVED, &cb_rcv3xx); osip_set_message_callback(osip, OSIP_ICT_STATUS_4XX_RECEIVED, &cb_rcv4xx); osip_set_message_callback(osip, OSIP_ICT_STATUS_5XX_RECEIVED, &cb_rcv5xx); osip_set_message_callback(osip, OSIP_ICT_STATUS_6XX_RECEIVED, &cb_rcv6xx); osip_set_message_callback(osip, OSIP_IST_STATUS_1XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_IST_STATUS_2XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_IST_STATUS_3XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_IST_STATUS_4XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_IST_STATUS_5XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_IST_STATUS_6XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_1XX_RECEIVED, &cb_rcv1xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_2XX_RECEIVED, &cb_rcv2xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_3XX_RECEIVED, &cb_rcv3xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_4XX_RECEIVED, &cb_rcv4xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_5XX_RECEIVED, &cb_rcv5xx); osip_set_message_callback(osip, OSIP_NICT_STATUS_6XX_RECEIVED, &cb_rcv6xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_1XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_2XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_3XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_4XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_5XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_STATUS_6XX_SENT, &cb_snd123456xx); osip_set_message_callback(osip, OSIP_NIST_CANCEL_RECEIVED, &cb_rcvcancel); osip_set_message_callback(osip, OSIP_NIST_REGISTER_RECEIVED, &cb_rcvregister); osip_set_message_callback(osip, OSIP_NIST_BYE_RECEIVED, &cb_rcvrequest); osip_set_message_callback(osip, OSIP_NIST_INFO_RECEIVED, &cb_rcvrequest); osip_set_message_callback(osip, OSIP_NIST_OPTIONS_RECEIVED, &cb_rcvrequest); osip_set_message_callback(osip, OSIP_NIST_SUBSCRIBE_RECEIVED, &cb_rcvrequest); osip_set_message_callback(osip, OSIP_NIST_NOTIFY_RECEIVED, &cb_rcvrequest); osip_set_message_callback(osip, OSIP_NIST_UNKNOWN_REQUEST_RECEIVED, &cb_rcvrequest); osip_set_transport_error_callback(osip, OSIP_ICT_TRANSPORT_ERROR, &cb_transport_error); osip_set_transport_error_callback(osip, OSIP_IST_TRANSPORT_ERROR, &cb_transport_error); osip_set_transport_error_callback(osip, OSIP_NICT_TRANSPORT_ERROR, &cb_transport_error); osip_set_transport_error_callback(osip, OSIP_NIST_TRANSPORT_ERROR, &cb_transport_error); #ifndef MINISIZE /* those methods are only used for log purpose except cb_transport_error which only apply to complete version */ osip_set_message_callback(osip, OSIP_ICT_STATUS_2XX_RECEIVED_AGAIN, &cb_rcvresp_retransmission); osip_set_message_callback(osip, OSIP_ICT_STATUS_3456XX_RECEIVED_AGAIN, &cb_rcvresp_retransmission); osip_set_message_callback(osip, OSIP_ICT_INVITE_SENT_AGAIN, &cb_sndreq_retransmission); osip_set_message_callback(osip, OSIP_IST_STATUS_2XX_SENT_AGAIN, &cb_sndresp_retransmission); osip_set_message_callback(osip, OSIP_IST_STATUS_3456XX_SENT_AGAIN, &cb_sndresp_retransmission); osip_set_message_callback(osip, OSIP_IST_INVITE_RECEIVED_AGAIN, &cb_rcvreq_retransmission); osip_set_message_callback(osip, OSIP_NICT_STATUS_2XX_RECEIVED_AGAIN, &cb_rcvresp_retransmission); osip_set_message_callback(osip, OSIP_NICT_STATUS_3456XX_RECEIVED_AGAIN, &cb_rcvresp_retransmission); osip_set_message_callback(osip, OSIP_NICT_REQUEST_SENT_AGAIN, &cb_sndreq_retransmission); osip_set_message_callback(osip, OSIP_NIST_STATUS_2XX_SENT_AGAIN, &cb_sndresp_retransmission); osip_set_message_callback(osip, OSIP_NIST_STATUS_3456XX_SENT_AGAIN, &cb_sndresp_retransmission); osip_set_message_callback(osip, OSIP_NIST_REQUEST_RECEIVED_AGAIN, &cb_rcvreq_retransmission); osip_set_message_callback(osip, OSIP_ICT_INVITE_SENT, &cb_sndinvite); osip_set_message_callback(osip, OSIP_ICT_ACK_SENT, &cb_sndack); osip_set_message_callback(osip, OSIP_NICT_REGISTER_SENT, &cb_sndregister); osip_set_message_callback(osip, OSIP_NICT_BYE_SENT, &cb_sndbye); osip_set_message_callback(osip, OSIP_NICT_CANCEL_SENT, &cb_sndcancel); osip_set_message_callback(osip, OSIP_NICT_INFO_SENT, &cb_sndinfo); osip_set_message_callback(osip, OSIP_NICT_OPTIONS_SENT, &cb_sndoptions); osip_set_message_callback(osip, OSIP_NICT_SUBSCRIBE_SENT, &cb_sndsubscribe); osip_set_message_callback(osip, OSIP_NICT_NOTIFY_SENT, &cb_sndnotify); osip_set_message_callback(osip, OSIP_NICT_UNKNOWN_REQUEST_SENT, &cb_sndunkrequest); osip_set_message_callback(osip, OSIP_IST_INVITE_RECEIVED, &cb_rcvinvite); osip_set_message_callback(osip, OSIP_IST_ACK_RECEIVED, &cb_rcvack); osip_set_message_callback(osip, OSIP_IST_ACK_RECEIVED_AGAIN, &cb_rcvack2); #endif return OSIP_SUCCESS; }