| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2015, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Joshua Colp <jcolp@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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \file
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \brief Core DNS Functionality | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author Joshua Colp <jcolp@digium.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*** MODULEINFO
 | 
					
						
							|  |  |  | 	<support_level>core</support_level> | 
					
						
							|  |  |  |  ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 04:33:33 -05:00
										 |  |  | #include "asterisk/_private.h"
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | #include "asterisk/linkedlists.h"
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | #include "asterisk/strings.h"
 | 
					
						
							|  |  |  | #include "asterisk/sched.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_core.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_srv.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_tlsa.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_recurring.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_resolver.h"
 | 
					
						
							|  |  |  | #include "asterisk/dns_internal.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | #include <netinet/in.h>
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | #include <arpa/nameser.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_RWLIST_HEAD_STATIC(resolvers, ast_dns_resolver); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_sched_context *sched; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_sched_context *ast_dns_get_sched(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return sched; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *ast_dns_query_get_name(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->name; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_query_get_rr_type(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->rr_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_query_get_rr_class(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->rr_class; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *ast_dns_query_get_data(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->user_data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_dns_result *ast_dns_query_get_result(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned int ast_dns_result_get_secure(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return result->secure; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned int ast_dns_result_get_bogus(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return result->bogus; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned int ast_dns_result_get_rcode(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return result->rcode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *ast_dns_result_get_canonical(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return result->canonical; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const struct ast_dns_record *ast_dns_result_get_records(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return AST_LIST_FIRST(&result->records); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *ast_dns_result_get_answer(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return result->answer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_result_get_lowest_ttl(const struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ttl = 0; | 
					
						
							|  |  |  | 	const struct ast_dns_record *record; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	if (ast_dns_result_get_rcode(result) == NXDOMAIN) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (record = ast_dns_result_get_records(result); record; record = ast_dns_record_get_next(record)) { | 
					
						
							|  |  |  | 		if (!ttl || (ast_dns_record_get_ttl(record) && (ast_dns_record_get_ttl(record) < ttl))) { | 
					
						
							|  |  |  | 			ttl = ast_dns_record_get_ttl(record); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ttl; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_dns_result_free(struct ast_dns_result *result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_record *record; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!result) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ((record = AST_LIST_REMOVE_HEAD(&result->records, list))) { | 
					
						
							|  |  |  | 		ast_free(record); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_free(result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_record_get_rr_type(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return record->rr_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_record_get_rr_class(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return record->rr_class; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_record_get_ttl(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return record->ttl; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *ast_dns_record_get_data(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | 	return record->data_ptr; | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | size_t ast_dns_record_get_data_size(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return record->data_len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | const struct ast_dns_record *ast_dns_record_get_next(const struct ast_dns_record *record) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return AST_LIST_NEXT(record, list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Destructor for an active DNS query */ | 
					
						
							|  |  |  | static void dns_query_active_destroy(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_query_active *active = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_cleanup(active->query); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief \brief Destructor for a DNS query */ | 
					
						
							|  |  |  | static void dns_query_destroy(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_query *query = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_cleanup(query->user_data); | 
					
						
							|  |  |  | 	ao2_cleanup(query->resolver_data); | 
					
						
							|  |  |  | 	ast_dns_result_free(query->result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | struct ast_dns_query *dns_query_alloc(const char *name, int rr_type, int rr_class, ast_dns_resolve_callback callback, void *data) | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 	struct ast_dns_query *query; | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(name)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution, no name provided\n"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_type > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution of '%s', resource record type '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			name, rr_type); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} else if (rr_type < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution of '%s', invalid resource record type '%d'\n", | 
					
						
							|  |  |  | 			name, rr_type); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_class > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution of '%s', resource record class '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			name, rr_class); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} else if (rr_class < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution of '%s', invalid resource class '%d'\n", | 
					
						
							|  |  |  | 			name, rr_class); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} else if (!callback) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform asynchronous resolution of '%s', no callback provided\n", | 
					
						
							|  |  |  | 			name); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 	query = ao2_alloc_options(sizeof(*query) + strlen(name) + 1, dns_query_destroy, AO2_ALLOC_OPT_LOCK_NOLOCK); | 
					
						
							|  |  |  | 	if (!query) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 	query->callback = callback; | 
					
						
							|  |  |  | 	query->user_data = ao2_bump(data); | 
					
						
							|  |  |  | 	query->rr_type = rr_type; | 
					
						
							|  |  |  | 	query->rr_class = rr_class; | 
					
						
							|  |  |  | 	strcpy(query->name, name); /* SAFE */ | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	AST_RWLIST_RDLOCK(&resolvers); | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 	query->resolver = AST_RWLIST_FIRST(&resolvers); | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 	AST_RWLIST_UNLOCK(&resolvers); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 	if (!query->resolver) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "Attempted to do a DNS query for '%s' of class '%d' and type '%d' but no resolver is available\n", | 
					
						
							|  |  |  | 			name, rr_class, rr_type); | 
					
						
							| 
									
										
										
										
											2015-04-13 10:47:01 -03:00
										 |  |  | 		ao2_ref(query, -1); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return query; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_dns_query_active *ast_dns_resolve_async(const char *name, int rr_type, int rr_class, ast_dns_resolve_callback callback, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_query_active *active; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	active = ao2_alloc_options(sizeof(*active), dns_query_active_destroy, AO2_ALLOC_OPT_LOCK_NOLOCK); | 
					
						
							|  |  |  | 	if (!active) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	active->query = dns_query_alloc(name, rr_type, rr_class, callback, data); | 
					
						
							|  |  |  | 	if (!active->query) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ao2_ref(active, -1); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (active->query->resolver->resolve(active->query)) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Resolver '%s' returned an error when resolving '%s' of class '%d' and type '%d'\n", | 
					
						
							|  |  |  | 			active->query->resolver->name, name, rr_class, rr_type); | 
					
						
							|  |  |  | 		ao2_ref(active, -1); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return active; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_resolve_cancel(struct ast_dns_query_active *active) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return active->query->resolver->cancel(active->query); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Structure used for signaling back for synchronous resolution completion */ | 
					
						
							|  |  |  | struct dns_synchronous_resolve { | 
					
						
							|  |  |  | 	/*! \brief Lock used for signaling */ | 
					
						
							|  |  |  | 	ast_mutex_t lock; | 
					
						
							|  |  |  | 	/*! \brief Condition used for signaling */ | 
					
						
							|  |  |  | 	ast_cond_t cond; | 
					
						
							|  |  |  | 	/*! \brief Whether the query has completed */ | 
					
						
							|  |  |  | 	unsigned int completed; | 
					
						
							|  |  |  | 	/*! \brief The result from the query */ | 
					
						
							|  |  |  | 	struct ast_dns_result *result; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Destructor for synchronous resolution structure */ | 
					
						
							|  |  |  | static void dns_synchronous_resolve_destroy(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dns_synchronous_resolve *synchronous = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_destroy(&synchronous->lock); | 
					
						
							|  |  |  | 	ast_cond_destroy(&synchronous->cond); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* This purposely does not unref result as it has been passed to the caller */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Callback used to implement synchronous resolution */ | 
					
						
							|  |  |  | static void dns_synchronous_resolve_callback(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dns_synchronous_resolve *synchronous = ast_dns_query_get_data(query); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	synchronous->result = query->result; | 
					
						
							|  |  |  | 	((struct ast_dns_query *)query)->result = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_lock(&synchronous->lock); | 
					
						
							|  |  |  | 	synchronous->completed = 1; | 
					
						
							|  |  |  | 	ast_cond_signal(&synchronous->cond); | 
					
						
							|  |  |  | 	ast_mutex_unlock(&synchronous->lock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_resolve(const char *name, int rr_type, int rr_class, struct ast_dns_result **result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dns_synchronous_resolve *synchronous; | 
					
						
							|  |  |  | 	struct ast_dns_query_active *active; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(name)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution, no name provided\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_type > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution of '%s', resource record type '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			name, rr_type); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (rr_type < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution of '%s', invalid resource record type '%d'\n", | 
					
						
							|  |  |  | 			name, rr_type); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_class > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution of '%s', resource record class '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			name, rr_class); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (rr_class < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution of '%s', invalid resource class '%d'\n", | 
					
						
							|  |  |  | 			name, rr_class); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (!result) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not perform synchronous resolution of '%s', no result pointer provided for storing results\n", | 
					
						
							|  |  |  | 			name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	synchronous = ao2_alloc_options(sizeof(*synchronous), dns_synchronous_resolve_destroy, AO2_ALLOC_OPT_LOCK_NOLOCK); | 
					
						
							|  |  |  | 	if (!synchronous) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_init(&synchronous->lock); | 
					
						
							|  |  |  | 	ast_cond_init(&synchronous->cond, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	active = ast_dns_resolve_async(name, rr_type, rr_class, dns_synchronous_resolve_callback, synchronous); | 
					
						
							|  |  |  | 	if (active) { | 
					
						
							|  |  |  | 		/* Wait for resolution to complete */ | 
					
						
							|  |  |  | 		ast_mutex_lock(&synchronous->lock); | 
					
						
							|  |  |  | 		while (!synchronous->completed) { | 
					
						
							|  |  |  | 			ast_cond_wait(&synchronous->cond, &synchronous->lock); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ast_mutex_unlock(&synchronous->lock); | 
					
						
							|  |  |  | 		ao2_ref(active, -1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*result = synchronous->result; | 
					
						
							|  |  |  | 	ao2_ref(synchronous, -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return *result ? 0 : -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_resolver_set_data(struct ast_dns_query *query, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (query->resolver_data) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	query->resolver_data = ao2_bump(data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *ast_dns_resolver_get_data(const struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return query->resolver_data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_dns_resolver_set_result(struct ast_dns_query *query, unsigned int secure, unsigned int bogus, | 
					
						
							|  |  |  | 	unsigned int rcode, const char *canonical, const char *answer, size_t answer_size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *buf_ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (secure && bogus) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not set result information, it can not be both secure and bogus\n", | 
					
						
							|  |  |  | 			query); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(canonical)) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not set result information since no canonical name was provided\n", | 
					
						
							|  |  |  | 			query); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 12:58:23 -05:00
										 |  |  | 	if (!answer) { | 
					
						
							|  |  |  | 		answer = ""; | 
					
						
							|  |  |  | 		answer_size = 0; | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Assuming zero-sized answer on NULL input\n", query); | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_dns_result_free(query->result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	query->result = ast_calloc(1, sizeof(*query->result) + strlen(canonical) + 1 + answer_size); | 
					
						
							|  |  |  | 	if (!query->result) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	query->result->secure = secure; | 
					
						
							|  |  |  | 	query->result->bogus = bogus; | 
					
						
							|  |  |  | 	query->result->rcode = rcode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	buf_ptr = query->result->buf; | 
					
						
							|  |  |  | 	strcpy(buf_ptr, canonical); /* SAFE */ | 
					
						
							|  |  |  | 	query->result->canonical = buf_ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	buf_ptr += strlen(canonical) + 1; | 
					
						
							|  |  |  | 	memcpy(buf_ptr, answer, answer_size); /* SAFE */ | 
					
						
							|  |  |  | 	query->result->answer = buf_ptr; | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | 	query->result->answer_size = answer_size; | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | static struct ast_dns_record *generic_record_alloc(struct ast_dns_query *query, const char *data, const size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_record *record; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	record = ast_calloc(1, sizeof(*record) + size); | 
					
						
							|  |  |  | 	if (!record) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	record->data_ptr = record->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return record; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | typedef struct ast_dns_record *(*dns_alloc_fn)(struct ast_dns_query *query, const char *data, const size_t size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static dns_alloc_fn dns_alloc_table [] = { | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	[T_NAPTR] = dns_naptr_alloc, | 
					
						
							|  |  |  | 	[T_SRV] = dns_srv_alloc, | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-18 23:48:52 -05:00
										 |  |  | static struct ast_dns_record *allocate_dns_record(unsigned int rr_type, struct ast_dns_query *query, const char *data, const size_t size) | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-12-18 23:48:52 -05:00
										 |  |  | 	dns_alloc_fn allocator = generic_record_alloc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rr_type < ARRAY_LEN(dns_alloc_table) && dns_alloc_table[rr_type]) { | 
					
						
							|  |  |  | 		allocator = dns_alloc_table[rr_type]; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return allocator(query, data, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | int ast_dns_resolver_add_record(struct ast_dns_query *query, int rr_type, int rr_class, int ttl, const char *data, const size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_record *record; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rr_type < 0) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not add record, invalid resource record type '%d'\n", | 
					
						
							|  |  |  | 			query, rr_type); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_type > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_debug(2, "Query '%p': Could not add record, resource record type '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			query, rr_type); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (rr_class < 0) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not add record, invalid resource record class '%d'\n", | 
					
						
							|  |  |  | 			query, rr_class); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	} else if (rr_class > 65536) { | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 		ast_debug(2, "Query '%p': Could not add record, resource record class '%d' exceeds maximum\n", | 
					
						
							|  |  |  | 			query, rr_class); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (ttl < 0) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not add record, invalid TTL '%d'\n", | 
					
						
							|  |  |  | 			query, ttl); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (!data || !size) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': Could not add record, no data specified\n", | 
					
						
							|  |  |  | 			query); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (!query->result) { | 
					
						
							|  |  |  | 		ast_debug(2, "Query '%p': No result was set on the query, thus records can not be added\n", | 
					
						
							|  |  |  | 			query); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | 	record = allocate_dns_record(rr_type, query, data, size); | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 	if (!record) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	record->rr_type = rr_type; | 
					
						
							|  |  |  | 	record->rr_class = rr_class; | 
					
						
							|  |  |  | 	record->ttl = ttl; | 
					
						
							|  |  |  | 	record->data_len = size; | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | 	memcpy(record->data_ptr, data, size); | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	AST_LIST_INSERT_TAIL(&query->result->records, record, list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | typedef void (*dns_sort_fn)(struct ast_dns_result *result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static dns_sort_fn dns_sort_table [] = { | 
					
						
							| 
									
										
										
										
											2016-11-20 09:19:18 +11:00
										 |  |  | 	[T_NAPTR] = dns_naptr_sort, | 
					
						
							|  |  |  | 	[T_SRV] = dns_srv_sort, | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void sort_result(int rr_type, struct ast_dns_result *result) | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | 	if (dns_sort_table[rr_type]) { | 
					
						
							|  |  |  | 		dns_sort_table[rr_type](result); | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_dns_resolver_completed(struct ast_dns_query *query) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sort_result(ast_dns_query_get_rr_type(query), query->result); | 
					
						
							| 
									
										
										
										
											2015-04-01 16:27:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | 	query->callback(query); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void dns_shutdown(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (sched) { | 
					
						
							|  |  |  | 		ast_sched_context_destroy(sched); | 
					
						
							|  |  |  | 		sched = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 04:33:33 -05:00
										 |  |  | int dns_core_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sched = ast_sched_context_create(); | 
					
						
							|  |  |  | 	if (!sched) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_sched_start_thread(sched)) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_register_cleanup(dns_shutdown); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-25 12:32:26 +00:00
										 |  |  | int ast_dns_resolver_register(struct ast_dns_resolver *resolver) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_resolver *iter; | 
					
						
							|  |  |  | 	int inserted = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!resolver) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (ast_strlen_zero(resolver->name)) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Registration of DNS resolver failed as it does not have a name\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (!resolver->resolve) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "DNS resolver '%s' does not implement the resolve callback which is required\n", | 
					
						
							|  |  |  | 			resolver->name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (!resolver->cancel) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "DNS resolver '%s' does not implement the cancel callback which is required\n", | 
					
						
							|  |  |  | 			resolver->name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_RWLIST_WRLOCK(&resolvers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE(&resolvers, iter, next) { | 
					
						
							|  |  |  | 		if (!strcmp(iter->name, resolver->name)) { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR, "A DNS resolver with the name '%s' is already registered\n", resolver->name); | 
					
						
							|  |  |  | 			AST_RWLIST_UNLOCK(&resolvers); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_RWLIST_TRAVERSE_SAFE_BEGIN(&resolvers, iter, next) { | 
					
						
							|  |  |  | 		if (iter->priority > resolver->priority) { | 
					
						
							|  |  |  | 			AST_RWLIST_INSERT_BEFORE_CURRENT(resolver, next); | 
					
						
							|  |  |  | 			inserted = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	AST_RWLIST_TRAVERSE_SAFE_END; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!inserted) { | 
					
						
							|  |  |  | 		AST_RWLIST_INSERT_TAIL(&resolvers, resolver, next); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_RWLIST_UNLOCK(&resolvers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_verb(2, "Registered DNS resolver '%s' with priority '%d'\n", resolver->name, resolver->priority); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ast_dns_resolver_unregister(struct ast_dns_resolver *resolver) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_dns_resolver *iter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!resolver) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_RWLIST_WRLOCK(&resolvers); | 
					
						
							|  |  |  | 	AST_RWLIST_TRAVERSE_SAFE_BEGIN(&resolvers, iter, next) { | 
					
						
							|  |  |  | 		if (resolver == iter) { | 
					
						
							|  |  |  | 			AST_RWLIST_REMOVE_CURRENT(next); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	AST_RWLIST_TRAVERSE_SAFE_END; | 
					
						
							|  |  |  | 	AST_RWLIST_UNLOCK(&resolvers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_verb(2, "Unregistered DNS resolver '%s'\n", resolver->name); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-04-09 14:58:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | char *dns_find_record(const char *record, size_t record_size, const char *response, size_t response_size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	size_t remaining_size = response_size; | 
					
						
							|  |  |  | 	const char *search_base = response; | 
					
						
							|  |  |  | 	char *record_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (1) { | 
					
						
							|  |  |  | 		record_offset = memchr(search_base, record[0], remaining_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ast_assert(record_offset != NULL); | 
					
						
							|  |  |  | 		ast_assert(search_base + remaining_size - record_offset >= record_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!memcmp(record_offset, record, record_size)) { | 
					
						
							|  |  |  | 			return record_offset; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		remaining_size -= record_offset - search_base; | 
					
						
							|  |  |  | 		search_base = record_offset + 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dns_parse_short(unsigned char *cur, uint16_t *val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* This assignment takes a big-endian 16-bit value and stores it in the
 | 
					
						
							|  |  |  | 	 * machine's native byte order. Using this method allows us to avoid potential | 
					
						
							|  |  |  | 	 * alignment issues in case the order is not on a short-addressable boundary. | 
					
						
							|  |  |  | 	 * See http://commandcenter.blogspot.com/2012/04/byte-order-fallacy.html for
 | 
					
						
							|  |  |  | 	 * more information | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	*val = (cur[1] << 0) | (cur[0] << 8); | 
					
						
							|  |  |  | 	return sizeof(*val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dns_parse_string(char *cur, uint8_t *size, char **val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	*size = *cur++; | 
					
						
							|  |  |  | 	*val = cur; | 
					
						
							|  |  |  | 	return *size + 1; | 
					
						
							|  |  |  | } |