| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2013, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Jonathan Rose <jrose@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 Parking Core | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author Jonathan Rose <jrose@digium.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/_private.h"
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | #include "asterisk/pbx.h"
 | 
					
						
							| 
									
										
										
										
											2013-07-25 04:06:32 +00:00
										 |  |  | #include "asterisk/bridge.h"
 | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | #include "asterisk/parking.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							| 
									
										
										
										
											2013-06-13 13:46:40 +00:00
										 |  |  | #include "asterisk/_private.h"
 | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | #include "asterisk/module.h"
 | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Message type for parked calls */ | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | STASIS_MESSAGE_TYPE_DEFN(ast_parked_call_type); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Topic for parking lots */ | 
					
						
							|  |  |  | static struct stasis_topic *parking_topic; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | /*! \brief The container for the parking provider */ | 
					
						
							|  |  |  | static AO2_GLOBAL_OBJ_STATIC(parking_provider); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 13:46:40 +00:00
										 |  |  | static void parking_stasis_cleanup(void) | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	STASIS_MESSAGE_TYPE_CLEANUP(ast_parked_call_type); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	ao2_cleanup(parking_topic); | 
					
						
							|  |  |  | 	parking_topic = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 13:46:40 +00:00
										 |  |  | int ast_parking_stasis_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (STASIS_MESSAGE_TYPE_INIT(ast_parked_call_type)) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	parking_topic = stasis_topic_create("ast_parking"); | 
					
						
							|  |  |  | 	if (!parking_topic) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_register_cleanup(parking_stasis_cleanup); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | struct stasis_topic *ast_parking_topic(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return parking_topic; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Destructor for parked_call_payload objects */ | 
					
						
							|  |  |  | static void parked_call_payload_destructor(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_parked_call_payload *park_obj = obj; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_cleanup(park_obj->parkee); | 
					
						
							| 
									
										
										
										
											2013-06-24 23:56:54 +00:00
										 |  |  | 	ao2_cleanup(park_obj->retriever); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	ast_string_field_free_memory(park_obj); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_parked_call_payload *ast_parked_call_payload_create(enum ast_parked_call_event_type event_type, | 
					
						
							| 
									
										
										
										
											2013-07-04 18:46:56 +00:00
										 |  |  | 		struct ast_channel_snapshot *parkee_snapshot, const char *parker_dial_string, | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 		struct ast_channel_snapshot *retriever_snapshot, const char *parkinglot, | 
					
						
							|  |  |  | 		unsigned int parkingspace, unsigned long int timeout, | 
					
						
							|  |  |  | 		unsigned long int duration) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_parked_call_payload *, payload, NULL, ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	payload = ao2_alloc(sizeof(*payload), parked_call_payload_destructor); | 
					
						
							|  |  |  | 	if (!payload) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_string_field_init(payload, 32)) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	payload->event_type = event_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_ref(parkee_snapshot, +1); | 
					
						
							|  |  |  | 	payload->parkee = parkee_snapshot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (retriever_snapshot) { | 
					
						
							|  |  |  | 		ao2_ref(retriever_snapshot, +1); | 
					
						
							|  |  |  | 		payload->retriever = retriever_snapshot; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (parkinglot) { | 
					
						
							|  |  |  | 		ast_string_field_set(payload, parkinglot, parkinglot); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-04 18:46:56 +00:00
										 |  |  | 	if (parker_dial_string) { | 
					
						
							|  |  |  | 		ast_string_field_set(payload, parker_dial_string, parker_dial_string); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	payload->parkingspace = parkingspace; | 
					
						
							|  |  |  | 	payload->timeout = timeout; | 
					
						
							|  |  |  | 	payload->duration = duration; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Bump the ref count by one since RAII_VAR is going to eat one when we leave. */ | 
					
						
							|  |  |  | 	ao2_ref(payload, +1); | 
					
						
							|  |  |  | 	return payload; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | int ast_parking_park_bridge_channel(struct ast_bridge_channel *parkee, const char *parkee_uuid, const char *parker_uuid, const char *app_data) | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, table, | 
					
						
							|  |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!table || !table->parking_park_bridge_channel) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (table->module_info) { | 
					
						
							|  |  |  | 		SCOPED_MODULE_USE(table->module_info->self); | 
					
						
							|  |  |  | 		return table->parking_park_bridge_channel(parkee, parkee_uuid, parker_uuid, app_data); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return table->parking_park_bridge_channel(parkee, parkee_uuid, parker_uuid, app_data); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | int ast_parking_blind_transfer_park(struct ast_bridge_channel *parker, const char *context, const char *exten) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, table, | 
					
						
							|  |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!table || !table->parking_blind_transfer_park) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (table->module_info) { | 
					
						
							|  |  |  | 		SCOPED_MODULE_USE(table->module_info->self); | 
					
						
							|  |  |  | 		return table->parking_blind_transfer_park(parker, context, exten); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return table->parking_blind_transfer_park(parker, context, exten); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_parking_park_call(struct ast_bridge_channel *parker, char *exten, size_t length) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, table, | 
					
						
							|  |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!table || !table->parking_park_call) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (table->module_info) { | 
					
						
							|  |  |  | 		SCOPED_MODULE_USE(table->module_info->self); | 
					
						
							|  |  |  | 		return table->parking_park_call(parker, exten, length); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return table->parking_park_call(parker, exten, length); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_parking_is_exten_park(const char *context, const char *exten) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, table, | 
					
						
							|  |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!table || !table->parking_is_exten_park) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (table->module_info) { | 
					
						
							|  |  |  | 		SCOPED_MODULE_USE(table->module_info->self); | 
					
						
							|  |  |  | 		return table->parking_is_exten_park(context, exten); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return table->parking_is_exten_park(context, exten); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | int ast_parking_register_bridge_features(struct ast_parking_bridge_feature_fn_table *fn_table) | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, wrapper, | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	if (fn_table->module_version != PARKING_MODULE_VERSION) { | 
					
						
							|  |  |  | 		ast_log(AST_LOG_WARNING, "Parking module provided incorrect parking module " | 
					
						
							|  |  |  | 			"version: %d (expected: %d)\n", fn_table->module_version, PARKING_MODULE_VERSION); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-07-26 21:34:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	if (wrapper) { | 
					
						
							|  |  |  | 		ast_log(AST_LOG_WARNING, "Parking provider already registered by %s!\n", | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 			wrapper->module_name); | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	wrapper = ao2_alloc(sizeof(*wrapper), NULL); | 
					
						
							|  |  |  | 	if (!wrapper) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 	*wrapper = *fn_table; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-06 19:26:48 +00:00
										 |  |  | 	ao2_global_obj_replace_unref(parking_provider, wrapper); | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | int ast_parking_unregister_bridge_features(const char *module_name) | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, wrapper, | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	if (!wrapper) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 	if (strcmp(wrapper->module_name, module_name)) { | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 		ast_log(AST_LOG_WARNING, "%s has not registered the parking provider\n", module_name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-06 19:26:48 +00:00
										 |  |  | 	ao2_global_obj_release(parking_provider); | 
					
						
							| 
									
										
										
										
											2013-08-01 20:55:17 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2013-05-21 18:00:22 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-08-17 15:01:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int ast_parking_provider_registered(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	RAII_VAR(struct ast_parking_bridge_feature_fn_table *, table, | 
					
						
							|  |  |  | 		ao2_global_obj_ref(parking_provider), ao2_cleanup); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return !!table; | 
					
						
							|  |  |  | } |