| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2013, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * David M. Lee, II <dlee@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 Stasis application command support. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author David M. Lee, II <dlee@digium.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "command.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/lock.h"
 | 
					
						
							|  |  |  | #include "asterisk/stasis_app_impl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct stasis_app_command { | 
					
						
							|  |  |  | 	ast_mutex_t lock; | 
					
						
							|  |  |  | 	ast_cond_t condition; | 
					
						
							|  |  |  | 	stasis_app_command_cb callback; | 
					
						
							|  |  |  | 	void *data; | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	command_data_destructor_fn data_destructor; | 
					
						
							| 
									
										
										
										
											2013-12-13 16:38:57 +00:00
										 |  |  | 	int retval; | 
					
						
							| 
									
										
										
										
											2019-10-23 12:36:17 -05:00
										 |  |  | 	unsigned int is_done:1; | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void command_dtor(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct stasis_app_command *command = obj; | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (command->data_destructor) { | 
					
						
							|  |  |  | 		command->data_destructor(command->data); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 	ast_mutex_destroy(&command->lock); | 
					
						
							|  |  |  | 	ast_cond_destroy(&command->condition); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct stasis_app_command *command_create( | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	stasis_app_command_cb callback, void *data, command_data_destructor_fn data_destructor) | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	struct stasis_app_command *command; | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	command = ao2_alloc(sizeof(*command), command_dtor); | 
					
						
							|  |  |  | 	if (!command) { | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 		if (data_destructor) { | 
					
						
							|  |  |  | 			data_destructor(data); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_init(&command->lock); | 
					
						
							|  |  |  | 	ast_cond_init(&command->condition, 0); | 
					
						
							|  |  |  | 	command->callback = callback; | 
					
						
							|  |  |  | 	command->data = data; | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	command->data_destructor = data_destructor; | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return command; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-13 16:38:57 +00:00
										 |  |  | void command_complete(struct stasis_app_command *command, int retval) | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-01-06 03:17:15 -05:00
										 |  |  | 	ast_mutex_lock(&command->lock); | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 	command->is_done = 1; | 
					
						
							|  |  |  | 	command->retval = retval; | 
					
						
							|  |  |  | 	ast_cond_signal(&command->condition); | 
					
						
							| 
									
										
										
										
											2018-01-06 03:17:15 -05:00
										 |  |  | 	ast_mutex_unlock(&command->lock); | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-13 16:38:57 +00:00
										 |  |  | int command_join(struct stasis_app_command *command) | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-01-06 03:17:15 -05:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_lock(&command->lock); | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 	while (!command->is_done) { | 
					
						
							|  |  |  | 		ast_cond_wait(&command->condition, &command->lock); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-06 03:17:15 -05:00
										 |  |  | 	ret = command->retval; | 
					
						
							|  |  |  | 	ast_mutex_unlock(&command->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void command_invoke(struct stasis_app_command *command, | 
					
						
							|  |  |  | 	struct stasis_app_control *control, struct ast_channel *chan) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-12-13 16:38:57 +00:00
										 |  |  | 	int retval = command->callback(control, chan, command->data); | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	if (command->data_destructor) { | 
					
						
							|  |  |  | 		command->data_destructor(command->data); | 
					
						
							|  |  |  | 		command->data_destructor = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-05-14 21:45:08 +00:00
										 |  |  | 	command_complete(command, retval); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | static void command_queue_prestart_destroy(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Clean up the container */ | 
					
						
							|  |  |  | 	ao2_cleanup(obj); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct ast_datastore_info command_queue_prestart = { | 
					
						
							| 
									
										
										
										
											2014-08-11 21:04:21 +00:00
										 |  |  | 	.type = "stasis-command-prestart-queue", | 
					
						
							|  |  |  | 	.destroy = command_queue_prestart_destroy, | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int command_prestart_queue_command(struct ast_channel *chan, | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 	stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor) | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_datastore *datastore; | 
					
						
							|  |  |  | 	struct ao2_container *command_queue; | 
					
						
							|  |  |  | 	RAII_VAR(struct stasis_app_command *, command, | 
					
						
							| 
									
										
										
										
											2014-08-22 16:52:51 +00:00
										 |  |  | 		command_create(command_fn, data, data_destructor), ao2_cleanup); | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!command) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 21:04:21 +00:00
										 |  |  | 	datastore = ast_channel_datastore_find(chan, &command_queue_prestart, NULL); | 
					
						
							|  |  |  | 	if (datastore) { | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | 		command_queue = datastore->data; | 
					
						
							|  |  |  | 		ao2_link(command_queue, command); | 
					
						
							| 
									
										
										
										
											2014-08-11 21:04:21 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 21:04:21 +00:00
										 |  |  | 	command_queue = ao2_container_alloc_list(AO2_ALLOC_OPT_LOCK_MUTEX, 0, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | 	if (!command_queue) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	datastore = ast_datastore_alloc(&command_queue_prestart, NULL); | 
					
						
							|  |  |  | 	if (!datastore) { | 
					
						
							|  |  |  | 		ao2_cleanup(command_queue); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_channel_datastore_add(chan, datastore); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	datastore->data = command_queue; | 
					
						
							|  |  |  | 	ao2_link(command_queue, command); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ao2_container *command_prestart_get_container(struct ast_channel *chan) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-08-11 21:04:21 +00:00
										 |  |  | 	struct ast_datastore *datastore = ast_channel_datastore_find(chan, &command_queue_prestart, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-07 15:30:19 +00:00
										 |  |  | 	if (!datastore) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ao2_bump(datastore->data); | 
					
						
							|  |  |  | } |