mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 06:26:41 +00:00 
			
		
		
		
	* Refactored pjproject code to support the new algorithms and
added a patch file to third-party/pjproject/patches
* Added new parameters to the pjsip auth object:
  * password_digest = <algorithm>:<digest>
  * supported_algorithms_uac = List of algorithms to support
    when acting as a UAC.
  * supported_algorithms_uas = List of algorithms to support
    when acting as a UAS.
  See the auth object in pjsip.conf.sample for detailed info.
* Updated both res_pjsip_authenticator_digest.c (for UAS) and
res_pjsip_outbound_authentocator_digest.c (UAC) to suport the
new algorithms.
The new algorithms are only available with the bundled version
of pjproject, or an external version > 2.14.1.  OpenSSL version
1.1.1 or greater is required to support SHA-512-256.
Resolves: #948
UserNote: The SHA-256 and SHA-512-256 algorithms are now available
for authentication as both a UAS and a UAC.
(cherry picked from commit 7dc9d85f2b)
		
	
		
			
				
	
	
		
			835 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			835 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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>
 | |
| #include <pjlib.h>
 | |
| 
 | |
| #include "asterisk/res_pjsip.h"
 | |
| #include "include/res_pjsip_private.h"
 | |
| #include "asterisk/pbx.h"
 | |
| #include "asterisk/sorcery.h"
 | |
| #include "asterisk/taskprocessor.h"
 | |
| #include "asterisk/ast_version.h"
 | |
| #include "asterisk/res_pjsip_cli.h"
 | |
| 
 | |
| #define DEFAULT_MAX_FORWARDS 70
 | |
| #define DEFAULT_KEEPALIVE_INTERVAL 90
 | |
| #define DEFAULT_USERAGENT_PREFIX "Asterisk PBX"
 | |
| #define DEFAULT_OUTBOUND_ENDPOINT "default_outbound_endpoint"
 | |
| #define DEFAULT_DEBUG "no"
 | |
| #define DEFAULT_ENDPOINT_IDENTIFIER_ORDER "ip,username,anonymous"
 | |
| #define DEFAULT_MAX_INITIAL_QUALIFY_TIME 0
 | |
| #define DEFAULT_FROM_USER "asterisk"
 | |
| #define DEFAULT_REALM "asterisk"
 | |
| #define DEFAULT_REGCONTEXT ""
 | |
| #define DEFAULT_CONTACT_EXPIRATION_CHECK_INTERVAL 30
 | |
| #define DEFAULT_DISABLE_MULTI_DOMAIN 0
 | |
| #define DEFAULT_VOICEMAIL_EXTENSION ""
 | |
| #define DEFAULT_UNIDENTIFIED_REQUEST_COUNT 5
 | |
| #define DEFAULT_UNIDENTIFIED_REQUEST_PERIOD 5
 | |
| #define DEFAULT_UNIDENTIFIED_REQUEST_PRUNE_INTERVAL 30
 | |
| #define DEFAULT_MWI_TPS_QUEUE_HIGH AST_TASKPROCESSOR_HIGH_WATER_LEVEL
 | |
| #define DEFAULT_MWI_TPS_QUEUE_LOW -1
 | |
| #define DEFAULT_MWI_DISABLE_INITIAL_UNSOLICITED 0
 | |
| #define DEFAULT_ALLOW_SENDING_180_AFTER_183 0
 | |
| #define DEFAULT_IGNORE_URI_USER_OPTIONS 0
 | |
| #define DEFAULT_USE_CALLERID_CONTACT 0
 | |
| #define DEFAULT_SEND_CONTACT_STATUS_ON_UPDATE_REGISTRATION 0
 | |
| #define DEFAULT_TASKPROCESSOR_OVERLOAD_TRIGGER TASKPROCESSOR_OVERLOAD_TRIGGER_GLOBAL
 | |
| #define DEFAULT_NOREFERSUB 1
 | |
| #define DEFAULT_ALL_CODECS_ON_EMPTY_REINVITE 0
 | |
| #define DEFAULT_AUTH_ALGORITHMS_UAS "MD5"
 | |
| #define DEFAULT_AUTH_ALGORITHMS_UAC "MD5"
 | |
| 
 | |
| /*!
 | |
|  * \brief Cached global config object
 | |
|  *
 | |
|  * \details
 | |
|  * Cached so we don't have to keep asking sorcery for the config.
 | |
|  * We could ask for it hundreds of times a second if not more.
 | |
|  */
 | |
| static AO2_GLOBAL_OBJ_STATIC(global_cfg);
 | |
| 
 | |
| static char default_useragent[256];
 | |
| 
 | |
| struct global_config {
 | |
| 	SORCERY_OBJECT(details);
 | |
| 	AST_DECLARE_STRING_FIELDS(
 | |
| 		AST_STRING_FIELD(useragent);
 | |
| 		AST_STRING_FIELD(regcontext);
 | |
| 		AST_STRING_FIELD(default_outbound_endpoint);
 | |
| 		/*! Debug logging yes|no|host */
 | |
| 		AST_STRING_FIELD(debug);
 | |
| 		/*! Order by which endpoint identifiers are checked (comma separated list) */
 | |
| 		AST_STRING_FIELD(endpoint_identifier_order);
 | |
| 		/*! User name to place in From header if there is no better option */
 | |
| 		AST_STRING_FIELD(default_from_user);
 | |
| 		/*! Default voicemail extension */
 | |
| 		AST_STRING_FIELD(default_voicemail_extension);
 | |
| 		/*! Realm to use in challenges before an endpoint is identified */
 | |
| 		AST_STRING_FIELD(default_realm);
 | |
| 		/*! Default authentication algorithms for UAS */
 | |
| 		AST_STRING_FIELD(default_auth_algorithms_uas);
 | |
| 		/*! Default authentication algorithms for UAC */
 | |
| 		AST_STRING_FIELD(default_auth_algorithms_uac);
 | |
| 	);
 | |
| 	/*! Value to put in Max-Forwards header */
 | |
| 	unsigned int max_forwards;
 | |
| 	/*! The interval at which to send keep alive messages to active connection-oriented transports */
 | |
| 	unsigned int keep_alive_interval;
 | |
| 	/*! The maximum time for all contacts to be qualified at startup */
 | |
| 	unsigned int max_initial_qualify_time;
 | |
| 	/*! The interval at which to check for expired contacts */
 | |
| 	unsigned int contact_expiration_check_interval;
 | |
| 	/*! Nonzero to disable multi domain support */
 | |
| 	unsigned int disable_multi_domain;
 | |
| 	/*! Nonzero to disable changing 180/SDP to 183/SDP */
 | |
| 	unsigned int allow_sending_180_after_183;
 | |
| 	/*! The maximum number of unidentified requests per source IP address before a security event is logged */
 | |
| 	unsigned int unidentified_request_count;
 | |
| 	/*! The period during which unidentified requests are accumulated */
 | |
| 	unsigned int unidentified_request_period;
 | |
| 	/*! Interval at which expired unidentified requests will be pruned */
 | |
| 	unsigned int unidentified_request_prune_interval;
 | |
| 	struct {
 | |
| 		/*! Taskprocessor high water alert trigger level */
 | |
| 		unsigned int tps_queue_high;
 | |
| 		/*! Taskprocessor low water clear alert level. */
 | |
| 		int tps_queue_low;
 | |
| 		/*! Nonzero to disable sending unsolicited mwi to all endpoints on startup */
 | |
| 		unsigned int disable_initial_unsolicited;
 | |
| 	} mwi;
 | |
| 	/*! Nonzero if URI user field options are ignored. */
 | |
| 	unsigned int ignore_uri_user_options;
 | |
| 	/*! Nonzero if CALLERID(num) is to be used as the default contact username instead of default_from_user */
 | |
| 	unsigned int use_callerid_contact;
 | |
| 	/*! Nonzero if need to send AMI ContactStatus event when a contact is updated */
 | |
| 	unsigned int send_contact_status_on_update_registration;
 | |
| 	/*! Trigger the distributor should use to pause accepting new dialogs */
 | |
| 	enum ast_sip_taskprocessor_overload_trigger overload_trigger;
 | |
| 	/*! Nonzero if norefersub is to be sent in Supported header */
 | |
| 	unsigned int norefersub;
 | |
| 	/*! Nonzero if we should return all codecs on empty re-INVITE */
 | |
| 	unsigned int all_codecs_on_empty_reinvite;
 | |
| };
 | |
| 
 | |
| static void global_destructor(void *obj)
 | |
| {
 | |
| 	struct global_config *cfg = obj;
 | |
| 
 | |
| 	ast_string_field_free_memory(cfg);
 | |
| }
 | |
| 
 | |
| static void *global_alloc(const char *name)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = ast_sorcery_generic_alloc(sizeof(*cfg), global_destructor);
 | |
| 	if (!cfg || ast_string_field_init(cfg, 100)) {
 | |
| 		ao2_cleanup(cfg);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return cfg;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * There is ever only one global section, so we can use a single global
 | |
|  * value here to track the regcontext through reloads.
 | |
|  */
 | |
| static char *previous_regcontext = NULL;
 | |
| 
 | |
| static int check_regcontext(const struct global_config *cfg)
 | |
| {
 | |
| 	char *current = NULL;
 | |
| 
 | |
| 	if (previous_regcontext && !strcmp(previous_regcontext, cfg->regcontext)) {
 | |
| 		/* Nothing changed so nothing to do */
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(cfg->regcontext)) {
 | |
| 		current = ast_strdup(cfg->regcontext);
 | |
| 		if (!current) {
 | |
| 			return -1;
 | |
| 		}
 | |
| 
 | |
| 		if (ast_sip_persistent_endpoint_add_to_regcontext(cfg->regcontext)) {
 | |
| 			ast_free(current);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(previous_regcontext)) {
 | |
| 		ast_context_destroy_by_name(previous_regcontext, "PJSIP");
 | |
| 		ast_free(previous_regcontext);
 | |
| 		previous_regcontext = NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (current) {
 | |
| 		previous_regcontext = current;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int global_apply(const struct ast_sorcery *sorcery, void *obj)
 | |
| {
 | |
| 	struct global_config *cfg = obj;
 | |
| 	char max_forwards[10];
 | |
| 	struct pjsip_auth_algorithm_type_vector algorithms;
 | |
| 	int res = 0;
 | |
| 
 | |
| 	if (ast_strlen_zero(cfg->debug)) {
 | |
| 		ast_log(LOG_ERROR,
 | |
| 			"Global option 'debug' can't be empty.  Set it to a valid value or remove the entry to accept 'no' as the default\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_strlen_zero(cfg->default_from_user)) {
 | |
| 		ast_log(LOG_ERROR,
 | |
| 			"Global option 'default_from_user' can't be empty.  Set it to a valid value or remove the entry to accept 'asterisk' as the default\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	snprintf(max_forwards, sizeof(max_forwards), "%u", cfg->max_forwards);
 | |
| 
 | |
| 	ast_sip_add_global_request_header("Max-Forwards", max_forwards, 1);
 | |
| 	ast_sip_add_global_request_header("User-Agent", cfg->useragent, 1);
 | |
| 	ast_sip_add_global_response_header("Server", cfg->useragent, 1);
 | |
| 
 | |
| 	if (check_regcontext(cfg)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	AST_VECTOR_INIT(&algorithms, 4);
 | |
| 	res = ast_sip_auth_digest_algorithms_vector_init("global",
 | |
| 		&algorithms, "UAS", cfg->default_auth_algorithms_uas);
 | |
| 	AST_VECTOR_FREE(&algorithms);
 | |
| 	if (res) {
 | |
| 		ast_log(LOG_WARNING, "global: Invalid values in default_auth_algorithms_uas. "
 | |
| 			"Defaulting to %s\n", DEFAULT_AUTH_ALGORITHMS_UAS);
 | |
| 		ast_string_field_set(cfg, default_auth_algorithms_uas, DEFAULT_AUTH_ALGORITHMS_UAS);
 | |
| 	}
 | |
| 	AST_VECTOR_INIT(&algorithms, 4);
 | |
| 	res = ast_sip_auth_digest_algorithms_vector_init("global",
 | |
| 		&algorithms, "UAC", cfg->default_auth_algorithms_uac);
 | |
| 	AST_VECTOR_FREE(&algorithms);
 | |
| 	if (res) {
 | |
| 		ast_log(LOG_WARNING, "global: Invalid values in default_auth_algorithms_uac. "
 | |
| 			"Defaulting to %s\n", DEFAULT_AUTH_ALGORITHMS_UAC);
 | |
| 		ast_string_field_set(cfg, default_auth_algorithms_uac, DEFAULT_AUTH_ALGORITHMS_UAC);
 | |
| 	}
 | |
| 
 | |
| 	ao2_t_global_obj_replace_unref(global_cfg, cfg, "Applying global settings");
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct global_config *get_global_cfg(void)
 | |
| {
 | |
| 	return ao2_global_obj_ref(global_cfg);
 | |
| }
 | |
| 
 | |
| char *ast_sip_global_default_outbound_endpoint(void)
 | |
| {
 | |
| 	char *str;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return ast_strdup(DEFAULT_OUTBOUND_ENDPOINT);
 | |
| 	}
 | |
| 
 | |
| 	str = ast_strdup(cfg->default_outbound_endpoint);
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return str;
 | |
| }
 | |
| 
 | |
| char *ast_sip_get_debug(void)
 | |
| {
 | |
| 	char *res;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return ast_strdup(DEFAULT_DEBUG);
 | |
| 	}
 | |
| 
 | |
| 	res = ast_strdup(cfg->debug);
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| char *ast_sip_get_regcontext(void)
 | |
| {
 | |
| 	char *res;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return ast_strdup(DEFAULT_REGCONTEXT);
 | |
| 	}
 | |
| 
 | |
| 	res = ast_strdup(cfg->regcontext);
 | |
| 	ao2_ref(cfg, -1);
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| char *ast_sip_get_default_voicemail_extension(void)
 | |
| {
 | |
| 	char *res;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return ast_strdup(DEFAULT_VOICEMAIL_EXTENSION);
 | |
| 	}
 | |
| 
 | |
| 	res = ast_strdup(cfg->default_voicemail_extension);
 | |
| 	ao2_ref(cfg, -1);
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| char *ast_sip_get_endpoint_identifier_order(void)
 | |
| {
 | |
| 	char *res;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return ast_strdup(DEFAULT_ENDPOINT_IDENTIFIER_ORDER);
 | |
| 	}
 | |
| 
 | |
| 	res = ast_strdup(cfg->endpoint_identifier_order);
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_keep_alive_interval(void)
 | |
| {
 | |
| 	unsigned int interval;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_KEEPALIVE_INTERVAL;
 | |
| 	}
 | |
| 
 | |
| 	interval = cfg->keep_alive_interval;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return interval;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_contact_expiration_check_interval(void)
 | |
| {
 | |
| 	unsigned int interval;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_CONTACT_EXPIRATION_CHECK_INTERVAL;
 | |
| 	}
 | |
| 
 | |
| 	interval = cfg->contact_expiration_check_interval;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return interval;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_disable_multi_domain(void)
 | |
| {
 | |
| 	unsigned int disable_multi_domain;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_DISABLE_MULTI_DOMAIN;
 | |
| 	}
 | |
| 
 | |
| 	disable_multi_domain = cfg->disable_multi_domain;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return disable_multi_domain;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_max_initial_qualify_time(void)
 | |
| {
 | |
| 	unsigned int time;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_MAX_INITIAL_QUALIFY_TIME;
 | |
| 	}
 | |
| 
 | |
| 	time = cfg->max_initial_qualify_time;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return time;
 | |
| }
 | |
| 
 | |
| void ast_sip_get_unidentified_request_thresholds(unsigned int *count, unsigned int *period,
 | |
| 	unsigned int *prune_interval)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		*count = DEFAULT_UNIDENTIFIED_REQUEST_COUNT;
 | |
| 		*period = DEFAULT_UNIDENTIFIED_REQUEST_PERIOD;
 | |
| 		*prune_interval = DEFAULT_UNIDENTIFIED_REQUEST_PRUNE_INTERVAL;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	*count = cfg->unidentified_request_count;
 | |
| 	*period = cfg->unidentified_request_period;
 | |
| 	*prune_interval = cfg->unidentified_request_prune_interval;
 | |
| 
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return;
 | |
| }
 | |
| 
 | |
| void ast_sip_get_default_realm(char *realm, size_t size)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		ast_copy_string(realm, DEFAULT_REALM, size);
 | |
| 	} else {
 | |
| 		ast_copy_string(realm, cfg->default_realm, size);
 | |
| 		ao2_ref(cfg, -1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_sip_get_default_auth_algorithms_uas(char *default_auth_algorithms_uas, size_t size)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		ast_copy_string(default_auth_algorithms_uas, DEFAULT_AUTH_ALGORITHMS_UAS, size);
 | |
| 	} else {
 | |
| 		ast_copy_string(default_auth_algorithms_uas, cfg->default_auth_algorithms_uas, size);
 | |
| 		ao2_ref(cfg, -1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_sip_get_default_auth_algorithms_uac(char *default_auth_algorithms_uac, size_t size)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		ast_copy_string(default_auth_algorithms_uac, DEFAULT_AUTH_ALGORITHMS_UAC, size);
 | |
| 	} else {
 | |
| 		ast_copy_string(default_auth_algorithms_uac, cfg->default_auth_algorithms_uac, size);
 | |
| 		ao2_ref(cfg, -1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ast_sip_get_default_from_user(char *from_user, size_t size)
 | |
| {
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		ast_copy_string(from_user, DEFAULT_FROM_USER, size);
 | |
| 	} else {
 | |
| 		ast_copy_string(from_user, cfg->default_from_user, size);
 | |
| 		ao2_ref(cfg, -1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| unsigned int ast_sip_get_mwi_tps_queue_high(void)
 | |
| {
 | |
| 	unsigned int tps_queue_high;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_MWI_TPS_QUEUE_HIGH;
 | |
| 	}
 | |
| 
 | |
| 	tps_queue_high = cfg->mwi.tps_queue_high;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return tps_queue_high;
 | |
| }
 | |
| 
 | |
| int ast_sip_get_mwi_tps_queue_low(void)
 | |
| {
 | |
| 	int tps_queue_low;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_MWI_TPS_QUEUE_LOW;
 | |
| 	}
 | |
| 
 | |
| 	tps_queue_low = cfg->mwi.tps_queue_low;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return tps_queue_low;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_mwi_disable_initial_unsolicited(void)
 | |
| {
 | |
| 	unsigned int disable_initial_unsolicited;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_MWI_DISABLE_INITIAL_UNSOLICITED;
 | |
| 	}
 | |
| 
 | |
| 	disable_initial_unsolicited = cfg->mwi.disable_initial_unsolicited;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return disable_initial_unsolicited;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_allow_sending_180_after_183(void)
 | |
| {
 | |
| 	unsigned int allow_sending_180_after_183;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_ALLOW_SENDING_180_AFTER_183;
 | |
| 	}
 | |
| 
 | |
| 	allow_sending_180_after_183 = cfg->allow_sending_180_after_183;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return allow_sending_180_after_183;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_ignore_uri_user_options(void)
 | |
| {
 | |
| 	unsigned int ignore_uri_user_options;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_IGNORE_URI_USER_OPTIONS;
 | |
| 	}
 | |
| 
 | |
| 	ignore_uri_user_options = cfg->ignore_uri_user_options;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return ignore_uri_user_options;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_use_callerid_contact(void)
 | |
| {
 | |
| 	unsigned int use_callerid_contact;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_USE_CALLERID_CONTACT;
 | |
| 	}
 | |
| 
 | |
| 	use_callerid_contact = cfg->use_callerid_contact;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return use_callerid_contact;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_send_contact_status_on_update_registration(void)
 | |
| {
 | |
| 	unsigned int send_contact_status_on_update_registration;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_SEND_CONTACT_STATUS_ON_UPDATE_REGISTRATION;
 | |
| 	}
 | |
| 
 | |
| 	send_contact_status_on_update_registration = cfg->send_contact_status_on_update_registration;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return send_contact_status_on_update_registration;
 | |
| }
 | |
| 
 | |
| enum ast_sip_taskprocessor_overload_trigger ast_sip_get_taskprocessor_overload_trigger(void)
 | |
| {
 | |
| 	enum ast_sip_taskprocessor_overload_trigger trigger;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_TASKPROCESSOR_OVERLOAD_TRIGGER;
 | |
| 	}
 | |
| 
 | |
| 	trigger = cfg->overload_trigger;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return trigger;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_norefersub(void)
 | |
| {
 | |
| 	unsigned int norefersub;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_NOREFERSUB;
 | |
| 	}
 | |
| 
 | |
| 	norefersub = cfg->norefersub;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return norefersub;
 | |
| }
 | |
| 
 | |
| unsigned int ast_sip_get_all_codecs_on_empty_reinvite(void)
 | |
| {
 | |
| 	unsigned int all_codecs_on_empty_reinvite;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	cfg = get_global_cfg();
 | |
| 	if (!cfg) {
 | |
| 		return DEFAULT_ALL_CODECS_ON_EMPTY_REINVITE;
 | |
| 	}
 | |
| 
 | |
| 	all_codecs_on_empty_reinvite = cfg->all_codecs_on_empty_reinvite;
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return all_codecs_on_empty_reinvite;
 | |
| }
 | |
| 
 | |
| static int overload_trigger_handler(const struct aco_option *opt,
 | |
| 	struct ast_variable *var, void *obj)
 | |
| {
 | |
| 	struct global_config *cfg = obj;
 | |
| 	if (!strcasecmp(var->value, "none")) {
 | |
| 		cfg->overload_trigger = TASKPROCESSOR_OVERLOAD_TRIGGER_NONE;
 | |
| 	} else if (!strcasecmp(var->value, "global")) {
 | |
| 		cfg->overload_trigger = TASKPROCESSOR_OVERLOAD_TRIGGER_GLOBAL;
 | |
| 	} else if (!strcasecmp(var->value, "pjsip_only")) {
 | |
| 		cfg->overload_trigger = TASKPROCESSOR_OVERLOAD_TRIGGER_PJSIP_ONLY;
 | |
| 	} else {
 | |
| 		ast_log(LOG_WARNING, "Unknown overload trigger '%s' specified for %s\n",
 | |
| 				var->value, var->name);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static const char *overload_trigger_map[] = {
 | |
| 	[TASKPROCESSOR_OVERLOAD_TRIGGER_NONE] = "none",
 | |
| 	[TASKPROCESSOR_OVERLOAD_TRIGGER_GLOBAL] = "global",
 | |
| 	[TASKPROCESSOR_OVERLOAD_TRIGGER_PJSIP_ONLY] = "pjsip_only"
 | |
| };
 | |
| 
 | |
| const char *ast_sip_overload_trigger_to_str(enum ast_sip_taskprocessor_overload_trigger trigger)
 | |
| {
 | |
| 	return ARRAY_IN_BOUNDS(trigger, overload_trigger_map) ?
 | |
| 		overload_trigger_map[trigger] : "";
 | |
| }
 | |
| 
 | |
| static int overload_trigger_to_str(const void *obj, const intptr_t *args, char **buf)
 | |
| {
 | |
| 	const struct global_config *cfg = obj;
 | |
| 	*buf = ast_strdup(ast_sip_overload_trigger_to_str(cfg->overload_trigger));
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*!
 | |
|  * \internal
 | |
|  * \brief Observer to set default global object if none exist.
 | |
|  *
 | |
|  * \param name Module name owning the sorcery instance.
 | |
|  * \param sorcery Instance being observed.
 | |
|  * \param object_type Name of object being observed.
 | |
|  * \param reloaded Non-zero if the object is being reloaded.
 | |
|  */
 | |
| static void global_loaded_observer(const char *name, const struct ast_sorcery *sorcery, const char *object_type, int reloaded)
 | |
| {
 | |
| 	struct ao2_container *globals;
 | |
| 	struct global_config *cfg;
 | |
| 
 | |
| 	if (strcmp(object_type, "global")) {
 | |
| 		/* Not interested */
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	globals = ast_sorcery_retrieve_by_fields(sorcery, "global",
 | |
| 		AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
 | |
| 	if (globals) {
 | |
| 		int count;
 | |
| 
 | |
| 		count = ao2_container_count(globals);
 | |
| 		ao2_ref(globals, -1);
 | |
| 
 | |
| 		if (1 < count) {
 | |
| 			ast_log(LOG_ERROR,
 | |
| 				"At most one pjsip.conf type=global object can be defined.  You have %d defined.\n",
 | |
| 				count);
 | |
| 			return;
 | |
| 		}
 | |
| 		if (count) {
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ast_debug(1, "No pjsip.conf type=global object exists so applying defaults.\n");
 | |
| 	cfg = ast_sorcery_alloc(sorcery, "global", NULL);
 | |
| 	if (!cfg) {
 | |
| 		return;
 | |
| 	}
 | |
| 	global_apply(sorcery, cfg);
 | |
| 	ao2_ref(cfg, -1);
 | |
| }
 | |
| 
 | |
| static const struct ast_sorcery_instance_observer observer_callbacks_global = {
 | |
| 	.object_type_loaded = global_loaded_observer,
 | |
| };
 | |
| 
 | |
| int sip_cli_print_global(struct ast_sip_cli_context *context)
 | |
| {
 | |
| 	struct global_config *cfg = get_global_cfg();
 | |
| 
 | |
| 	if (!cfg) {
 | |
| 		cfg = ast_sorcery_alloc(ast_sip_get_sorcery(), "global", NULL);
 | |
| 		if (!cfg) {
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ast_str_append(&context->output_buffer, 0, "\nGlobal Settings:\n\n");
 | |
| 	ast_sip_cli_print_sorcery_objectset(cfg, context, 0);
 | |
| 
 | |
| 	ao2_ref(cfg, -1);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int ast_sip_destroy_sorcery_global(void)
 | |
| {
 | |
| 	struct ast_sorcery *sorcery = ast_sip_get_sorcery();
 | |
| 
 | |
| 	ast_sorcery_instance_observer_remove(sorcery, &observer_callbacks_global);
 | |
| 
 | |
| 	if (previous_regcontext) {
 | |
| 		ast_context_destroy_by_name(previous_regcontext, "PJSIP");
 | |
| 		ast_free(previous_regcontext);
 | |
| 	}
 | |
| 
 | |
| 	ao2_t_global_obj_release(global_cfg, "Module is unloading");
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int ast_sip_initialize_sorcery_global(void)
 | |
| {
 | |
| 	struct ast_sorcery *sorcery = ast_sip_get_sorcery();
 | |
| 
 | |
| 	snprintf(default_useragent, sizeof(default_useragent), "%s %s",
 | |
| 		DEFAULT_USERAGENT_PREFIX, ast_get_version());
 | |
| 
 | |
| 	ast_sorcery_apply_default(sorcery, "global", "config", "pjsip.conf,criteria=type=global,single_object=yes,explicit_name=global");
 | |
| 
 | |
| 	if (ast_sorcery_object_register(sorcery, "global", global_alloc, NULL, global_apply)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "type", "", OPT_NOOP_T, 0, 0);
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "max_forwards",
 | |
| 		__stringify(DEFAULT_MAX_FORWARDS),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, max_forwards));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "user_agent", default_useragent,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, useragent));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_outbound_endpoint",
 | |
| 		DEFAULT_OUTBOUND_ENDPOINT,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, default_outbound_endpoint));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "debug", DEFAULT_DEBUG,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, debug));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "endpoint_identifier_order",
 | |
| 		DEFAULT_ENDPOINT_IDENTIFIER_ORDER,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, endpoint_identifier_order));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "keep_alive_interval",
 | |
| 		__stringify(DEFAULT_KEEPALIVE_INTERVAL),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, keep_alive_interval));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "max_initial_qualify_time",
 | |
| 		__stringify(DEFAULT_MAX_INITIAL_QUALIFY_TIME),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, max_initial_qualify_time));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_from_user", DEFAULT_FROM_USER,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, default_from_user));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_voicemail_extension",
 | |
| 		DEFAULT_VOICEMAIL_EXTENSION, OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config,
 | |
| 		default_voicemail_extension));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "regcontext", DEFAULT_REGCONTEXT,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, regcontext));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "contact_expiration_check_interval",
 | |
| 		__stringify(DEFAULT_CONTACT_EXPIRATION_CHECK_INTERVAL),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, contact_expiration_check_interval));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "disable_multi_domain",
 | |
| 		DEFAULT_DISABLE_MULTI_DOMAIN ? "yes" : "no",
 | |
| 		OPT_BOOL_T, 1, FLDSET(struct global_config, disable_multi_domain));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "unidentified_request_count",
 | |
| 		__stringify(DEFAULT_UNIDENTIFIED_REQUEST_COUNT),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, unidentified_request_count));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "unidentified_request_period",
 | |
| 		__stringify(DEFAULT_UNIDENTIFIED_REQUEST_PERIOD),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, unidentified_request_period));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "unidentified_request_prune_interval",
 | |
| 		__stringify(DEFAULT_UNIDENTIFIED_REQUEST_PRUNE_INTERVAL),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, unidentified_request_prune_interval));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_realm", DEFAULT_REALM,
 | |
| 		OPT_STRINGFIELD_T, 0, STRFLDSET(struct global_config, default_realm));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "mwi_tps_queue_high",
 | |
| 		__stringify(DEFAULT_MWI_TPS_QUEUE_HIGH),
 | |
| 		OPT_UINT_T, 0, FLDSET(struct global_config, mwi.tps_queue_high));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "mwi_tps_queue_low",
 | |
| 		__stringify(DEFAULT_MWI_TPS_QUEUE_LOW),
 | |
| 		OPT_INT_T, 0, FLDSET(struct global_config, mwi.tps_queue_low));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "mwi_disable_initial_unsolicited",
 | |
| 		DEFAULT_MWI_DISABLE_INITIAL_UNSOLICITED ? "yes" : "no",
 | |
| 		OPT_BOOL_T, 1, FLDSET(struct global_config, mwi.disable_initial_unsolicited));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "allow_sending_180_after_183",
 | |
| 		DEFAULT_ALLOW_SENDING_180_AFTER_183 ? "yes" : "no",
 | |
| 		OPT_BOOL_T, 1, FLDSET(struct global_config, allow_sending_180_after_183));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "ignore_uri_user_options",
 | |
| 		DEFAULT_IGNORE_URI_USER_OPTIONS ? "yes" : "no",
 | |
| 		OPT_BOOL_T, 1, FLDSET(struct global_config, ignore_uri_user_options));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "use_callerid_contact",
 | |
| 		DEFAULT_USE_CALLERID_CONTACT ? "yes" : "no",
 | |
| 		OPT_YESNO_T, 1, FLDSET(struct global_config, use_callerid_contact));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "send_contact_status_on_update_registration",
 | |
| 		DEFAULT_SEND_CONTACT_STATUS_ON_UPDATE_REGISTRATION ? "yes" : "no",
 | |
| 		OPT_YESNO_T, 1, FLDSET(struct global_config, send_contact_status_on_update_registration));
 | |
| 	ast_sorcery_object_field_register_custom(sorcery, "global", "taskprocessor_overload_trigger",
 | |
| 		overload_trigger_map[DEFAULT_TASKPROCESSOR_OVERLOAD_TRIGGER],
 | |
| 		overload_trigger_handler, overload_trigger_to_str, NULL, 0, 0);
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "norefersub",
 | |
| 		DEFAULT_NOREFERSUB ? "yes" : "no",
 | |
| 		OPT_YESNO_T, 1, FLDSET(struct global_config, norefersub));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "all_codecs_on_empty_reinvite",
 | |
| 		DEFAULT_ALL_CODECS_ON_EMPTY_REINVITE ? "yes" : "no",
 | |
| 		OPT_BOOL_T, 1, FLDSET(struct global_config, all_codecs_on_empty_reinvite));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_auth_algorithms_uas",
 | |
| 		DEFAULT_AUTH_ALGORITHMS_UAS, OPT_STRINGFIELD_T, 0,
 | |
| 		STRFLDSET(struct global_config, default_auth_algorithms_uas));
 | |
| 	ast_sorcery_object_field_register(sorcery, "global", "default_auth_algorithms_uac",
 | |
| 		DEFAULT_AUTH_ALGORITHMS_UAC, OPT_STRINGFIELD_T, 0,
 | |
| 		STRFLDSET(struct global_config, default_auth_algorithms_uac));
 | |
| 
 | |
| 	if (ast_sorcery_instance_observer_add(sorcery, &observer_callbacks_global)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 	ast_sorcery_load_object(ast_sip_get_sorcery(), "global");
 | |
| 
 | |
| 	return 0;
 | |
| }
 |