| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2013, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Mark Michelson <mmichelson@digium.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * See http://www.asterisk.org for more information about
 | 
					
						
							|  |  |  |  * the Asterisk project. Please do not directly contact | 
					
						
							|  |  |  |  * any of the maintainers of this project for assistance; | 
					
						
							|  |  |  |  * the project provides a web site, mailing lists and IRC | 
					
						
							|  |  |  |  * channels for your use. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <pjsip.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-30 18:14:50 +00:00
										 |  |  | #include "asterisk/res_pjsip.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | #include "include/res_pjsip_private.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | #include "asterisk/taskprocessor.h"
 | 
					
						
							|  |  |  | #include "asterisk/threadpool.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int distribute(void *data); | 
					
						
							|  |  |  | static pj_bool_t distributor(pjsip_rx_data *rdata); | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | static pj_status_t record_serializer(pjsip_tx_data *tdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static pjsip_module distributor_mod = { | 
					
						
							|  |  |  | 	.name = {"Request Distributor", 19}, | 
					
						
							|  |  |  | 	.priority = PJSIP_MOD_PRIORITY_TSX_LAYER - 6, | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	.on_tx_request = record_serializer, | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	.on_rx_request = distributor, | 
					
						
							|  |  |  | 	.on_rx_response = distributor, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | /*!
 | 
					
						
							|  |  |  |  * \internal | 
					
						
							|  |  |  |  * \brief Record the task's serializer name on the tdata structure. | 
					
						
							|  |  |  |  * \since 14.0.0 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param tdata The outgoing message. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \retval PJ_SUCCESS. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static pj_status_t record_serializer(pjsip_tx_data *tdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_taskprocessor *serializer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	serializer = ast_threadpool_serializer_get_current(); | 
					
						
							|  |  |  | 	if (serializer) { | 
					
						
							|  |  |  | 		const char *name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		name = ast_taskprocessor_name(serializer); | 
					
						
							|  |  |  | 		if (!ast_strlen_zero(name) | 
					
						
							|  |  |  | 			&& (!tdata->mod_data[distributor_mod.id] | 
					
						
							|  |  |  | 				|| strcmp(tdata->mod_data[distributor_mod.id], name))) { | 
					
						
							|  |  |  | 			char *tdata_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* The serializer in use changed. */ | 
					
						
							|  |  |  | 			tdata_name = pj_pool_alloc(tdata->pool, strlen(name) + 1); | 
					
						
							|  |  |  | 			strcpy(tdata_name, name);/* Safe */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			tdata->mod_data[distributor_mod.id] = tdata_name; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return PJ_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*!
 | 
					
						
							|  |  |  |  * \internal | 
					
						
							|  |  |  |  * \brief Find the request tdata to get the serializer it used. | 
					
						
							|  |  |  |  * \since 14.0.0 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param rdata The incoming message. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \retval serializer on success. | 
					
						
							|  |  |  |  * \retval NULL on error or could not find the serializer. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct ast_taskprocessor *find_request_serializer(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_taskprocessor *serializer = NULL; | 
					
						
							|  |  |  | 	pj_str_t tsx_key; | 
					
						
							|  |  |  | 	pjsip_transaction *tsx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pjsip_tsx_create_key(rdata->tp_info.pool, &tsx_key, PJSIP_ROLE_UAC, | 
					
						
							|  |  |  | 		&rdata->msg_info.cseq->method, rdata); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tsx = pjsip_tsx_layer_find_tsx(&tsx_key, PJ_TRUE); | 
					
						
							|  |  |  | 	if (!tsx) { | 
					
						
							|  |  |  | 		ast_debug(1, "Could not find %.*s transaction for %d response.\n", | 
					
						
							|  |  |  | 			(int) pj_strlen(&rdata->msg_info.cseq->method.name), | 
					
						
							|  |  |  | 			pj_strbuf(&rdata->msg_info.cseq->method.name), | 
					
						
							|  |  |  | 			rdata->msg_info.msg->line.status.code); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (tsx->last_tx) { | 
					
						
							|  |  |  | 		const char *serializer_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		serializer_name = tsx->last_tx->mod_data[distributor_mod.id]; | 
					
						
							|  |  |  | 		if (!ast_strlen_zero(serializer_name)) { | 
					
						
							|  |  |  | 			serializer = ast_taskprocessor_get(serializer_name, TPS_REF_IF_EXISTS); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAVE_PJ_TRANSACTION_GRP_LOCK
 | 
					
						
							|  |  |  | 	pj_grp_lock_release(tsx->grp_lock); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	pj_mutex_unlock(tsx->mutex); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return serializer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | /*! Dialog-specific information the distributor uses */ | 
					
						
							|  |  |  | struct distributor_dialog_data { | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	/*! Serializer to distribute tasks to for this dialog */ | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	struct ast_taskprocessor *serializer; | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	/*! Endpoint associated with this dialog */ | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	struct ast_sip_endpoint *endpoint; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*!
 | 
					
						
							|  |  |  |  * \internal | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Call this with the dialog locked | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct distributor_dialog_data *distributor_dialog_data_alloc(pjsip_dialog *dlg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct distributor_dialog_data *dist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dist = PJ_POOL_ZALLOC_T(dlg->pool, struct distributor_dialog_data); | 
					
						
							|  |  |  | 	pjsip_dlg_set_mod_data(dlg, distributor_mod.id, dist); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return dist; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_sip_dialog_set_serializer(pjsip_dialog *dlg, struct ast_taskprocessor *serializer) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct distributor_dialog_data *dist; | 
					
						
							|  |  |  | 	SCOPED_LOCK(lock, dlg, pjsip_dlg_inc_lock, pjsip_dlg_dec_lock); | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	dist = pjsip_dlg_get_mod_data(dlg, distributor_mod.id); | 
					
						
							|  |  |  | 	if (!dist) { | 
					
						
							|  |  |  | 		dist = distributor_dialog_data_alloc(dlg); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dist->serializer = serializer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_sip_dialog_set_endpoint(pjsip_dialog *dlg, struct ast_sip_endpoint *endpoint) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct distributor_dialog_data *dist; | 
					
						
							|  |  |  | 	SCOPED_LOCK(lock, dlg, pjsip_dlg_inc_lock, pjsip_dlg_dec_lock); | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	dist = pjsip_dlg_get_mod_data(dlg, distributor_mod.id); | 
					
						
							|  |  |  | 	if (!dist) { | 
					
						
							|  |  |  | 		dist = distributor_dialog_data_alloc(dlg); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dist->endpoint = endpoint; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_sip_endpoint *ast_sip_dialog_get_endpoint(pjsip_dialog *dlg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct distributor_dialog_data *dist; | 
					
						
							|  |  |  | 	SCOPED_LOCK(lock, dlg, pjsip_dlg_inc_lock, pjsip_dlg_dec_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dist = pjsip_dlg_get_mod_data(dlg, distributor_mod.id); | 
					
						
							|  |  |  | 	if (!dist || !dist->endpoint) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ao2_ref(dist->endpoint, +1); | 
					
						
							|  |  |  | 	return dist->endpoint; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-28 15:43:15 +00:00
										 |  |  | static pjsip_dialog *find_dialog(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	pj_str_t tsx_key; | 
					
						
							|  |  |  | 	pjsip_transaction *tsx; | 
					
						
							|  |  |  | 	pjsip_dialog *dlg; | 
					
						
							|  |  |  | 	pj_str_t *local_tag; | 
					
						
							|  |  |  | 	pj_str_t *remote_tag; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-06 00:31:48 +00:00
										 |  |  | 	if (!rdata->msg_info.msg) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-28 15:43:15 +00:00
										 |  |  | 	if (rdata->msg_info.msg->type == PJSIP_REQUEST_MSG) { | 
					
						
							|  |  |  | 		local_tag = &rdata->msg_info.to->tag; | 
					
						
							|  |  |  | 		remote_tag = &rdata->msg_info.from->tag; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		local_tag = &rdata->msg_info.from->tag; | 
					
						
							|  |  |  | 		remote_tag = &rdata->msg_info.to->tag; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* We can only call the convenient method for
 | 
					
						
							|  |  |  | 	 *  1) responses | 
					
						
							|  |  |  | 	 *  2) non-CANCEL requests | 
					
						
							|  |  |  | 	 *  3) CANCEL requests with a to-tag | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (rdata->msg_info.msg->type == PJSIP_RESPONSE_MSG || | 
					
						
							|  |  |  | 			pjsip_method_cmp(&rdata->msg_info.msg->line.req.method, &pjsip_cancel_method) || | 
					
						
							|  |  |  | 			rdata->msg_info.to->tag.slen != 0) { | 
					
						
							|  |  |  | 		return pjsip_ua_find_dialog(&rdata->msg_info.cid->id, local_tag, | 
					
						
							|  |  |  | 				remote_tag, PJ_TRUE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Incoming CANCEL without a to-tag can't use same method for finding the
 | 
					
						
							|  |  |  | 	 * dialog. Instead, we have to find the matching INVITE transaction and | 
					
						
							|  |  |  | 	 * then get the dialog from the transaction | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	pjsip_tsx_create_key(rdata->tp_info.pool, &tsx_key, PJSIP_ROLE_UAS, | 
					
						
							|  |  |  | 			pjsip_get_invite_method(), rdata); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tsx = pjsip_tsx_layer_find_tsx(&tsx_key, PJ_TRUE); | 
					
						
							|  |  |  | 	if (!tsx) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Could not find matching INVITE transaction for CANCEL request\n"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dlg = pjsip_tsx_get_dlg(tsx); | 
					
						
							| 
									
										
										
										
											2013-11-16 13:51:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAVE_PJ_TRANSACTION_GRP_LOCK
 | 
					
						
							|  |  |  | 	pj_grp_lock_release(tsx->grp_lock); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-08-28 15:43:15 +00:00
										 |  |  | 	pj_mutex_unlock(tsx->mutex); | 
					
						
							| 
									
										
										
										
											2013-11-16 13:51:04 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-08-28 15:43:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!dlg) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pjsip_dlg_inc_lock(dlg); | 
					
						
							|  |  |  | 	return dlg; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-06 18:21:12 +00:00
										 |  |  | static pj_bool_t endpoint_lookup(pjsip_rx_data *rdata); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static pjsip_module endpoint_mod = { | 
					
						
							|  |  |  | 	.name = {"Endpoint Identifier", 19}, | 
					
						
							|  |  |  | 	.priority = PJSIP_MOD_PRIORITY_TSX_LAYER - 3, | 
					
						
							|  |  |  | 	.on_rx_request = endpoint_lookup, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | static pj_bool_t distributor(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-28 15:43:15 +00:00
										 |  |  | 	pjsip_dialog *dlg = find_dialog(rdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	struct distributor_dialog_data *dist = NULL; | 
					
						
							|  |  |  | 	struct ast_taskprocessor *serializer = NULL; | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	struct ast_taskprocessor *req_serializer = NULL; | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	pjsip_rx_data *clone; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dlg) { | 
					
						
							|  |  |  | 		dist = pjsip_dlg_get_mod_data(dlg, distributor_mod.id); | 
					
						
							|  |  |  | 		if (dist) { | 
					
						
							|  |  |  | 			serializer = dist->serializer; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	if (serializer) { | 
					
						
							|  |  |  | 		/* We have a serializer so we know where to send the message. */ | 
					
						
							|  |  |  | 	} else if (rdata->msg_info.msg->type == PJSIP_RESPONSE_MSG) { | 
					
						
							|  |  |  | 		req_serializer = find_request_serializer(rdata); | 
					
						
							|  |  |  | 		serializer = req_serializer; | 
					
						
							|  |  |  | 	} else if (!pjsip_method_cmp(&rdata->msg_info.msg->line.req.method, &pjsip_cancel_method) | 
					
						
							|  |  |  | 		|| !pjsip_method_cmp(&rdata->msg_info.msg->line.req.method, &pjsip_bye_method)) { | 
					
						
							|  |  |  | 		/* We have a BYE or CANCEL request without a serializer. */ | 
					
						
							|  |  |  | 		pjsip_endpt_respond_stateless(ast_sip_get_pjsip_endpoint(), rdata, | 
					
						
							|  |  |  | 			PJSIP_SC_CALL_TSX_DOES_NOT_EXIST, NULL, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-25 00:44:24 +00:00
										 |  |  | 		goto end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pjsip_rx_data_clone(rdata, 0, &clone); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dist) { | 
					
						
							| 
									
										
										
										
											2014-11-06 18:21:12 +00:00
										 |  |  | 		clone->endpt_info.mod_data[endpoint_mod.id] = ao2_bump(dist->endpoint); | 
					
						
							| 
									
										
										
										
											2013-07-25 00:44:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	ast_sip_push_task(serializer, distribute, clone); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-25 00:44:24 +00:00
										 |  |  | end: | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	if (dlg) { | 
					
						
							|  |  |  | 		pjsip_dlg_dec_lock(dlg); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-06-05 15:37:33 -05:00
										 |  |  | 	ast_taskprocessor_unreference(req_serializer); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return PJ_TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | static struct ast_sip_auth *artificial_auth; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int create_artificial_auth(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!(artificial_auth = ast_sorcery_alloc( | 
					
						
							|  |  |  | 		      ast_sip_get_sorcery(), SIP_SORCERY_AUTH_TYPE, "artificial"))) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Unable to create artificial auth\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_string_field_set(artificial_auth, realm, "asterisk"); | 
					
						
							|  |  |  | 	ast_string_field_set(artificial_auth, auth_user, ""); | 
					
						
							|  |  |  | 	ast_string_field_set(artificial_auth, auth_pass, ""); | 
					
						
							|  |  |  | 	artificial_auth->type = AST_SIP_AUTH_TYPE_ARTIFICIAL; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_sip_auth *ast_sip_get_artificial_auth(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_ref(artificial_auth, +1); | 
					
						
							|  |  |  | 	return artificial_auth; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | static struct ast_sip_endpoint *artificial_endpoint = NULL; | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int create_artificial_endpoint(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!(artificial_endpoint = ast_sorcery_alloc( | 
					
						
							|  |  |  | 		      ast_sip_get_sorcery(), "endpoint", NULL))) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-09 16:10:05 +00:00
										 |  |  | 	AST_VECTOR_INIT(&artificial_endpoint->inbound_auths, 1); | 
					
						
							| 
									
										
										
										
											2014-01-14 16:43:33 +00:00
										 |  |  | 	/* Pushing a bogus value into the vector will ensure that
 | 
					
						
							|  |  |  | 	 * the proper size of the vector is returned. This value is | 
					
						
							|  |  |  | 	 * not actually used anywhere | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | 	AST_VECTOR_APPEND(&artificial_endpoint->inbound_auths, ast_strdup("artificial-auth")); | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_sip_endpoint *ast_sip_get_artificial_endpoint(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_ref(artificial_endpoint, +1); | 
					
						
							|  |  |  | 	return artificial_endpoint; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-25 17:41:38 +00:00
										 |  |  | static void log_unidentified_request(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char from_buf[PJSIP_MAX_URL_SIZE]; | 
					
						
							|  |  |  | 	char callid_buf[PJSIP_MAX_URL_SIZE]; | 
					
						
							|  |  |  | 	pjsip_uri_print(PJSIP_URI_IN_FROMTO_HDR, rdata->msg_info.from->uri, from_buf, PJSIP_MAX_URL_SIZE); | 
					
						
							|  |  |  | 	ast_copy_pj_str(callid_buf, &rdata->msg_info.cid->id, PJSIP_MAX_URL_SIZE); | 
					
						
							|  |  |  | 	ast_log(LOG_NOTICE, "Request from '%s' failed for '%s:%d' (callid: %s) - No matching endpoint found\n", | 
					
						
							|  |  |  | 		from_buf, rdata->pkt_info.src_name, rdata->pkt_info.src_port, callid_buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | static pj_bool_t endpoint_lookup(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_sip_endpoint *endpoint; | 
					
						
							|  |  |  | 	int is_ack = rdata->msg_info.msg->line.req.method.id == PJSIP_ACK_METHOD; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-06 18:21:12 +00:00
										 |  |  | 	endpoint = rdata->endpt_info.mod_data[endpoint_mod.id]; | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	if (endpoint) { | 
					
						
							| 
									
										
										
										
											2014-11-06 18:21:12 +00:00
										 |  |  | 		return PJ_FALSE; | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-06 18:21:12 +00:00
										 |  |  | 	endpoint = ast_sip_identify_endpoint(rdata); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	if (!endpoint && !is_ack) { | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 		char name[AST_UUID_STR_LEN] = ""; | 
					
						
							|  |  |  | 		pjsip_uri *from = rdata->msg_info.from->uri; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 		/* always use an artificial endpoint - per discussion no reason
 | 
					
						
							|  |  |  | 		   to have "alwaysauthreject" as an option.  It is felt using it | 
					
						
							|  |  |  | 		   was a bug fix and it is not needed since we are not worried about | 
					
						
							|  |  |  | 		   breaking old stuff and we really don't want to enable the discovery | 
					
						
							|  |  |  | 		   of SIP accounts */ | 
					
						
							|  |  |  | 		endpoint = ast_sip_get_artificial_endpoint(); | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (PJSIP_URI_SCHEME_IS_SIP(from) || PJSIP_URI_SCHEME_IS_SIPS(from)) { | 
					
						
							|  |  |  | 			pjsip_sip_uri *sip_from = pjsip_uri_get_uri(from); | 
					
						
							|  |  |  | 			ast_copy_pj_str(name, &sip_from->user, sizeof(name)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-25 17:41:38 +00:00
										 |  |  | 		log_unidentified_request(rdata); | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 		ast_sip_report_invalid_endpoint(name, rdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	rdata->endpt_info.mod_data[endpoint_mod.id] = endpoint; | 
					
						
							|  |  |  | 	return PJ_FALSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static pj_bool_t authenticate(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_sip_endpoint *, endpoint, ast_pjsip_rdata_get_endpoint(rdata), ao2_cleanup); | 
					
						
							|  |  |  | 	int is_ack = rdata->msg_info.msg->line.req.method.id == PJSIP_ACK_METHOD; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_assert(endpoint != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!is_ack && ast_sip_requires_authentication(endpoint, rdata)) { | 
					
						
							|  |  |  | 		pjsip_tx_data *tdata; | 
					
						
							|  |  |  | 		pjsip_endpt_create_response(ast_sip_get_pjsip_endpoint(), rdata, 401, NULL, &tdata); | 
					
						
							|  |  |  | 		switch (ast_sip_check_authentication(endpoint, rdata, tdata)) { | 
					
						
							|  |  |  | 		case AST_SIP_AUTHENTICATION_CHALLENGE: | 
					
						
							|  |  |  | 			/* Send the 401 we created for them */ | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 			ast_sip_report_auth_challenge_sent(endpoint, rdata, tdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 			pjsip_endpt_send_response2(ast_sip_get_pjsip_endpoint(), rdata, tdata, NULL, NULL); | 
					
						
							|  |  |  | 			return PJ_TRUE; | 
					
						
							|  |  |  | 		case AST_SIP_AUTHENTICATION_SUCCESS: | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 			ast_sip_report_auth_success(endpoint, rdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 			pjsip_tx_data_dec_ref(tdata); | 
					
						
							|  |  |  | 			return PJ_FALSE; | 
					
						
							|  |  |  | 		case AST_SIP_AUTHENTICATION_FAILED: | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 			ast_sip_report_auth_failed_challenge_response(endpoint, rdata); | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 			pjsip_endpt_send_response2(ast_sip_get_pjsip_endpoint(), rdata, tdata, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 			return PJ_TRUE; | 
					
						
							|  |  |  | 		case AST_SIP_AUTHENTICATION_ERROR: | 
					
						
							| 
									
										
										
										
											2013-06-22 14:03:22 +00:00
										 |  |  | 			ast_sip_report_auth_failed_challenge_response(endpoint, rdata); | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 			pjsip_tx_data_dec_ref(tdata); | 
					
						
							|  |  |  | 			pjsip_endpt_respond_stateless(ast_sip_get_pjsip_endpoint(), rdata, 500, NULL, NULL, NULL); | 
					
						
							|  |  |  | 			return PJ_TRUE; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return PJ_FALSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static pjsip_module auth_mod = { | 
					
						
							|  |  |  | 	.name = {"Request Authenticator", 21}, | 
					
						
							| 
									
										
										
										
											2014-01-15 13:16:10 +00:00
										 |  |  | 	.priority = PJSIP_MOD_PRIORITY_APPLICATION - 2, | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	.on_rx_request = authenticate, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int distribute(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static pjsip_process_rdata_param param = { | 
					
						
							|  |  |  | 		.start_mod = &distributor_mod, | 
					
						
							|  |  |  | 		.idx_after_start = 1, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	pj_bool_t handled; | 
					
						
							|  |  |  | 	pjsip_rx_data *rdata = data; | 
					
						
							|  |  |  | 	int is_request = rdata->msg_info.msg->type == PJSIP_REQUEST_MSG; | 
					
						
							|  |  |  | 	int is_ack = is_request ? rdata->msg_info.msg->line.req.method.id == PJSIP_ACK_METHOD : 0; | 
					
						
							|  |  |  | 	struct ast_sip_endpoint *endpoint; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pjsip_endpt_process_rx_data(ast_sip_get_pjsip_endpoint(), rdata, ¶m, &handled); | 
					
						
							|  |  |  | 	if (!handled && is_request && !is_ack) { | 
					
						
							|  |  |  | 		pjsip_endpt_respond_stateless(ast_sip_get_pjsip_endpoint(), rdata, 501, NULL, NULL, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* The endpoint_mod stores an endpoint reference in the mod_data of rdata. This
 | 
					
						
							|  |  |  | 	 * is the only appropriate spot to actually decrement the reference. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	endpoint = rdata->endpt_info.mod_data[endpoint_mod.id]; | 
					
						
							|  |  |  | 	ao2_cleanup(endpoint); | 
					
						
							|  |  |  | 	pjsip_rx_data_free_cloned(rdata); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_sip_endpoint *ast_pjsip_rdata_get_endpoint(pjsip_rx_data *rdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_sip_endpoint *endpoint = rdata->endpt_info.mod_data[endpoint_mod.id]; | 
					
						
							|  |  |  | 	if (endpoint) { | 
					
						
							|  |  |  | 		ao2_ref(endpoint, +1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return endpoint; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_sip_initialize_distributor(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 	if (create_artificial_endpoint() || create_artificial_auth()) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | 	if (internal_sip_register_service(&distributor_mod)) { | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | 	if (internal_sip_register_service(&endpoint_mod)) { | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | 	if (internal_sip_register_service(&auth_mod)) { | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 18:25:31 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void ast_sip_destroy_distributor(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-01-27 19:12:56 +00:00
										 |  |  | 	internal_sip_unregister_service(&distributor_mod); | 
					
						
							|  |  |  | 	internal_sip_unregister_service(&endpoint_mod); | 
					
						
							|  |  |  | 	internal_sip_unregister_service(&auth_mod); | 
					
						
							| 
									
										
										
										
											2013-07-02 17:06:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ao2_cleanup(artificial_auth); | 
					
						
							|  |  |  | 	ao2_cleanup(artificial_endpoint); | 
					
						
							|  |  |  | } |