| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2005, Anthony Minessale II | 
					
						
							| 
									
										
										
										
											2006-01-03 22:16:23 +00:00
										 |  |  |  * Copyright (C) 2005 - 2006, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Mark Spencer <markster@digium.com> | 
					
						
							|  |  |  |  * Kevin P. Fleming <kpfleming@digium.com> | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * Based on app_muxmon.c provided by | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * Anthony Minessale II <anthmct@yahoo.com> | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * This program is free software, distributed under the terms of | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-31 15:33:27 +00:00
										 |  |  |  * \brief MixMonitor() - Record a call and mix the audio during the recording | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup applications | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \author Mark Spencer <markster@digium.com> | 
					
						
							|  |  |  |  * \author Kevin P. Fleming <kpfleming@digium.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Based on app_muxmon.c provided by | 
					
						
							|  |  |  |  * Anthony Minessale II <anthmct@yahoo.com> | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2005-11-08 04:48:00 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2005-11-08 04:48:00 +00:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/file.h"
 | 
					
						
							|  |  |  | #include "asterisk/logger.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | #include "asterisk/audiohook.h"
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/lock.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/options.h"
 | 
					
						
							|  |  |  | #include "asterisk/app.h"
 | 
					
						
							|  |  |  | #include "asterisk/linkedlists.h"
 | 
					
						
							| 
									
										
										
										
											2006-01-13 03:34:31 +00:00
										 |  |  | #include "asterisk/utils.h"
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | #define get_volfactor(x) x ? ((x > 0) ? (1 << x) : ((1 << abs(x)) * -1)) : 0
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static const char *app = "MixMonitor"; | 
					
						
							|  |  |  | static const char *synopsis = "Record a call and mix the audio during the recording"; | 
					
						
							|  |  |  | static const char *desc = "" | 
					
						
							|  |  |  | "  MixMonitor(<file>.<ext>[|<options>[|<command>]])\n\n" | 
					
						
							|  |  |  | "Records the audio on the current channel to the specified file.\n" | 
					
						
							|  |  |  | "If the filename is an absolute path, uses that path, otherwise\n" | 
					
						
							|  |  |  | "creates the file in the configured monitoring directory from\n" | 
					
						
							|  |  |  | "asterisk.conf.\n\n" | 
					
						
							|  |  |  | "Valid options:\n" | 
					
						
							|  |  |  | " a      - Append to the file instead of overwriting it.\n" | 
					
						
							|  |  |  | " b      - Only save audio to the file while the channel is bridged.\n" | 
					
						
							| 
									
										
										
										
											2007-01-25 01:51:35 +00:00
										 |  |  | "          Note: Does not include conferences or sounds played to each bridged\n" | 
					
						
							|  |  |  | "                party.\n" | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | " v(<x>) - Adjust the heard volume by a factor of <x> (range -4 to 4)\n"	 | 
					
						
							|  |  |  | " V(<x>) - Adjust the spoken volume by a factor of <x> (range -4 to 4)\n"	 | 
					
						
							|  |  |  | " W(<x>) - Adjust the both heard and spoken volumes by a factor of <x>\n" | 
					
						
							|  |  |  | "         (range -4 to 4)\n\n"	 | 
					
						
							|  |  |  | "<command> will be executed when the recording is over\n" | 
					
						
							| 
									
										
										
										
											2007-06-27 23:25:12 +00:00
										 |  |  | "Any strings matching ^{X} will be unescaped to ${X}.\n" | 
					
						
							|  |  |  | "All variables will be evaluated at the time MixMonitor is called.\n" | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | "The variable MIXMONITOR_FILENAME will contain the filename used to record.\n" | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | ""; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | static const char *stop_app = "StopMixMonitor"; | 
					
						
							|  |  |  | static const char *stop_synopsis = "Stop recording a call through MixMonitor"; | 
					
						
							|  |  |  | static const char *stop_desc = "" | 
					
						
							|  |  |  | "  StopMixMonitor()\n\n" | 
					
						
							|  |  |  | "Stops the audio recording that was started with a call to MixMonitor()\n" | 
					
						
							|  |  |  | "on the current channel.\n" | 
					
						
							|  |  |  | ""; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-14 14:08:19 +00:00
										 |  |  | struct module_symbols *me; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static const char *mixmonitor_spy_type = "MixMonitor"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct mixmonitor { | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	struct ast_audiohook audiohook; | 
					
						
							| 
									
										
										
										
											2007-01-22 19:13:44 +00:00
										 |  |  | 	char *filename; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	char *post_process; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	char *name; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	unsigned int flags; | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 	struct mixmonitor_ds *mixmonitor_ds; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | enum { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 	MUXFLAG_APPEND = (1 << 1), | 
					
						
							|  |  |  | 	MUXFLAG_BRIDGED = (1 << 2), | 
					
						
							|  |  |  | 	MUXFLAG_VOLUME = (1 << 3), | 
					
						
							|  |  |  | 	MUXFLAG_READVOLUME = (1 << 4), | 
					
						
							|  |  |  | 	MUXFLAG_WRITEVOLUME = (1 << 5), | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | } mixmonitor_flags; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | enum { | 
					
						
							|  |  |  | 	OPT_ARG_READVOLUME = 0, | 
					
						
							|  |  |  | 	OPT_ARG_WRITEVOLUME, | 
					
						
							|  |  |  | 	OPT_ARG_VOLUME, | 
					
						
							|  |  |  | 	OPT_ARG_ARRAY_SIZE, | 
					
						
							|  |  |  | } mixmonitor_args; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_APP_OPTIONS(mixmonitor_opts, { | 
					
						
							|  |  |  | 	AST_APP_OPTION('a', MUXFLAG_APPEND), | 
					
						
							|  |  |  | 	AST_APP_OPTION('b', MUXFLAG_BRIDGED), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('v', MUXFLAG_READVOLUME, OPT_ARG_READVOLUME), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('V', MUXFLAG_WRITEVOLUME, OPT_ARG_WRITEVOLUME), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('W', MUXFLAG_VOLUME, OPT_ARG_VOLUME), | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | /* This structure is used as a means of making sure that our pointer to
 | 
					
						
							|  |  |  |  * the channel we are monitoring remains valid. This is very similar to  | 
					
						
							|  |  |  |  * what is used in app_chanspy.c. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct mixmonitor_ds { | 
					
						
							|  |  |  | 	struct ast_channel *chan; | 
					
						
							|  |  |  | 	/* These condition variables are used to be sure that the channel
 | 
					
						
							|  |  |  | 	 * hangup code completes before the mixmonitor thread attempts to | 
					
						
							|  |  |  | 	 * free this structure. The combination of a bookean flag and a | 
					
						
							|  |  |  | 	 * ast_cond_t ensure that no matter what order the threads run in, | 
					
						
							|  |  |  | 	 * we are guaranteed to never have the waiting thread block forever | 
					
						
							|  |  |  | 	 * in the case that the signaling thread runs first. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	unsigned int destruction_ok; | 
					
						
							|  |  |  | 	ast_cond_t destruction_condition; | 
					
						
							|  |  |  | 	ast_mutex_t lock; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mixmonitor_ds_destroy(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mixmonitor_ds *mixmonitor_ds = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_lock(&mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	mixmonitor_ds->chan = NULL; | 
					
						
							|  |  |  | 	mixmonitor_ds->destruction_ok = 1; | 
					
						
							|  |  |  | 	ast_cond_signal(&mixmonitor_ds->destruction_condition); | 
					
						
							|  |  |  | 	ast_mutex_unlock(&mixmonitor_ds->lock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mixmonitor_ds_chan_fixup(void *data, struct ast_channel *old_chan, struct ast_channel *new_chan) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mixmonitor_ds *mixmonitor_ds = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_lock(&mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	mixmonitor_ds->chan = new_chan; | 
					
						
							|  |  |  | 	ast_mutex_unlock(&mixmonitor_ds->lock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_datastore_info mixmonitor_ds_info = { | 
					
						
							|  |  |  | 	.type = "mixmonitor", | 
					
						
							|  |  |  | 	.destroy = mixmonitor_ds_destroy, | 
					
						
							|  |  |  | 	.chan_fixup = mixmonitor_ds_chan_fixup, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | static int startmon(struct ast_channel *chan, struct ast_audiohook *audiohook)  | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *peer; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!chan) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	res = ast_audiohook_attach(chan, audiohook); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!res && ast_test_flag(chan, AST_FLAG_NBRIDGE) && (peer = ast_bridged_channel(chan))) | 
					
						
							|  |  |  | 		ast_softhangup(peer, AST_SOFTHANGUP_UNBRIDGE);	 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | #define SAMPLES_PER_FRAME 160
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static void *mixmonitor_thread(void *obj)  | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mixmonitor *mixmonitor = obj; | 
					
						
							| 
									
										
										
										
											2007-01-22 19:13:44 +00:00
										 |  |  | 	struct ast_filestream *fs = NULL; | 
					
						
							|  |  |  | 	unsigned int oflags; | 
					
						
							|  |  |  | 	char *ext; | 
					
						
							|  |  |  | 	int errflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (option_verbose > 1) | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		ast_verbose(VERBOSE_PREFIX_2 "Begin MixMonitor Recording %s\n", mixmonitor->name); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	ast_audiohook_lock(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2006-03-04 11:51:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	while (mixmonitor->audiohook.status == AST_AUDIOHOOK_STATUS_RUNNING) { | 
					
						
							|  |  |  | 		struct ast_frame *fr = NULL; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		ast_audiohook_trigger_wait(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 		if (mixmonitor->audiohook.status != AST_AUDIOHOOK_STATUS_RUNNING) | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 		if (!(fr = ast_audiohook_read_frame(&mixmonitor->audiohook, SAMPLES_PER_FRAME, AST_AUDIOHOOK_DIRECTION_BOTH, AST_FORMAT_SLINEAR))) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		ast_mutex_lock(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							|  |  |  | 		if (!ast_test_flag(mixmonitor, MUXFLAG_BRIDGED) || (mixmonitor->mixmonitor_ds->chan && ast_bridged_channel(mixmonitor->mixmonitor_ds->chan))) { | 
					
						
							|  |  |  | 			ast_mutex_unlock(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 			/* Initialize the file if not already done so */ | 
					
						
							|  |  |  | 			if (!fs && !errflag) { | 
					
						
							|  |  |  | 				oflags = O_CREAT | O_WRONLY; | 
					
						
							|  |  |  | 				oflags |= ast_test_flag(mixmonitor, MUXFLAG_APPEND) ? O_APPEND : O_TRUNC; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if ((ext = strrchr(mixmonitor->filename, '.'))) | 
					
						
							|  |  |  | 					*(ext++) = '\0'; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					ext = "raw"; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (!(fs = ast_writefile(mixmonitor->filename, ext, NULL, oflags, 0, 0644))) { | 
					
						
							|  |  |  | 					ast_log(LOG_ERROR, "Cannot open %s.%s\n", mixmonitor->filename, ext); | 
					
						
							|  |  |  | 					errflag = 1; | 
					
						
							| 
									
										
										
										
											2007-01-22 19:13:44 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-18 01:51:21 +00:00
										 |  |  | 			/* Write out the frame */ | 
					
						
							|  |  |  | 			if (fs) | 
					
						
							|  |  |  | 				ast_writestream(fs, fr); | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			ast_mutex_unlock(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 		/* All done! free it. */ | 
					
						
							|  |  |  | 		ast_frame_free(fr, 0); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-03 17:25:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	ast_audiohook_detach(&mixmonitor->audiohook); | 
					
						
							|  |  |  | 	ast_audiohook_unlock(&mixmonitor->audiohook); | 
					
						
							|  |  |  | 	ast_audiohook_destroy(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2006-09-03 23:30:37 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (option_verbose > 1) | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		ast_verbose(VERBOSE_PREFIX_2 "End MixMonitor Recording %s\n", mixmonitor->name); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-26 17:20:58 +00:00
										 |  |  | 	if (fs) | 
					
						
							|  |  |  | 		ast_closestream(fs); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	if (mixmonitor->post_process) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		if (option_verbose > 2) | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 			ast_verbose(VERBOSE_PREFIX_2 "Executing [%s]\n", mixmonitor->post_process); | 
					
						
							|  |  |  | 		ast_safe_system(mixmonitor->post_process); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 	ast_mutex_lock(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	if (!mixmonitor->mixmonitor_ds->destruction_ok) { | 
					
						
							|  |  |  | 		ast_cond_wait(&mixmonitor->mixmonitor_ds->destruction_condition, &mixmonitor->mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_mutex_unlock(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							| 
									
										
										
										
											2009-02-05 18:47:24 +00:00
										 |  |  | 	ast_mutex_destroy(&mixmonitor->mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	ast_cond_destroy(&mixmonitor->mixmonitor_ds->destruction_condition); | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 	ast_free(mixmonitor->mixmonitor_ds); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	free(mixmonitor); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | static int setup_mixmonitor_ds(struct mixmonitor *mixmonitor, struct ast_channel *chan) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_datastore *datastore = NULL; | 
					
						
							|  |  |  | 	struct mixmonitor_ds *mixmonitor_ds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(mixmonitor_ds = ast_calloc(1, sizeof(*mixmonitor_ds)))) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	ast_mutex_init(&mixmonitor_ds->lock); | 
					
						
							|  |  |  | 	ast_cond_init(&mixmonitor_ds->destruction_condition, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(datastore = ast_channel_datastore_alloc(&mixmonitor_ds_info, NULL))) { | 
					
						
							|  |  |  | 		ast_free(mixmonitor_ds); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* No need to lock mixmonitor_ds since this is still operating in the channel's thread */ | 
					
						
							|  |  |  | 	mixmonitor_ds->chan = chan; | 
					
						
							|  |  |  | 	datastore->data = mixmonitor_ds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_channel_lock(chan); | 
					
						
							|  |  |  | 	ast_channel_datastore_add(chan, datastore); | 
					
						
							|  |  |  | 	ast_channel_unlock(chan); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mixmonitor->mixmonitor_ds = mixmonitor_ds; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static void launch_monitor_thread(struct ast_channel *chan, const char *filename, unsigned int flags, | 
					
						
							|  |  |  | 				  int readvol, int writevol, const char *post_process)  | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	pthread_attr_t attr; | 
					
						
							|  |  |  | 	pthread_t thread; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	struct mixmonitor *mixmonitor; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	char postprocess2[1024] = ""; | 
					
						
							|  |  |  | 	size_t len; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-01-22 19:13:44 +00:00
										 |  |  | 	len = sizeof(*mixmonitor) + strlen(chan->name) + strlen(filename) + 2; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* If a post process system command is given attach it to the structure */ | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(post_process)) { | 
					
						
							|  |  |  | 		char *p1, *p2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p1 = ast_strdupa(post_process); | 
					
						
							|  |  |  | 		for (p2 = p1; *p2 ; p2++) { | 
					
						
							|  |  |  | 			if (*p2 == '^' && *(p2+1) == '{') { | 
					
						
							|  |  |  | 				*p2 = '$'; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		pbx_substitute_variables_helper(chan, p1, postprocess2, sizeof(postprocess2) - 1); | 
					
						
							|  |  |  | 		if (!ast_strlen_zero(postprocess2)) | 
					
						
							|  |  |  | 			len += strlen(postprocess2) + 1; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	/* Pre-allocate mixmonitor structure and spy */ | 
					
						
							|  |  |  | 	if (!(mixmonitor = calloc(1, len))) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	/* Copy over flags and channel name */ | 
					
						
							|  |  |  | 	mixmonitor->flags = flags; | 
					
						
							| 
									
										
										
										
											2009-02-05 17:34:33 +00:00
										 |  |  | 	if (setup_mixmonitor_ds(mixmonitor, chan)) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	mixmonitor->name = (char *) mixmonitor + sizeof(*mixmonitor); | 
					
						
							|  |  |  | 	strcpy(mixmonitor->name, chan->name); | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(postprocess2)) { | 
					
						
							| 
									
										
										
										
											2007-01-29 23:43:40 +00:00
										 |  |  | 		mixmonitor->post_process = mixmonitor->name + strlen(mixmonitor->name) + strlen(filename) + 2; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 		strcpy(mixmonitor->post_process, postprocess2); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-01-22 19:13:44 +00:00
										 |  |  | 	mixmonitor->filename = (char *) mixmonitor + sizeof(*mixmonitor) + strlen(chan->name) + 1; | 
					
						
							|  |  |  | 	strcpy(mixmonitor->filename, filename); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	/* Setup the actual spy before creating our thread */ | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	if (ast_audiohook_init(&mixmonitor->audiohook, AST_AUDIOHOOK_TYPE_SPY, mixmonitor_spy_type)) { | 
					
						
							|  |  |  | 		free(mixmonitor); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-03-12 18:26:37 +00:00
										 |  |  | 	ast_set_flag(&mixmonitor->audiohook, AST_AUDIOHOOK_TRIGGER_SYNC); | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (readvol) | 
					
						
							|  |  |  | 		mixmonitor->audiohook.options.read_volume = readvol; | 
					
						
							|  |  |  | 	if (writevol) | 
					
						
							|  |  |  | 		mixmonitor->audiohook.options.write_volume = writevol; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	if (startmon(chan, &mixmonitor->audiohook)) { | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to add '%s' spy to channel '%s'\n", | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 			mixmonitor_spy_type, chan->name); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		/* Since we couldn't add ourselves - bail out! */ | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 		ast_audiohook_destroy(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		free(mixmonitor); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	pthread_attr_init(&attr); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); | 
					
						
							| 
									
										
										
										
											2006-10-04 19:47:22 +00:00
										 |  |  | 	ast_pthread_create_background(&thread, &attr, mixmonitor_thread, mixmonitor); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	pthread_attr_destroy(&attr); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static int mixmonitor_exec(struct ast_channel *chan, void *data) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	int x, readvol = 0, writevol = 0; | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	struct ast_module_user *u; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	struct ast_flags flags = {0}; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	char *parse; | 
					
						
							|  |  |  | 	AST_DECLARE_APP_ARGS(args, | 
					
						
							|  |  |  | 		AST_APP_ARG(filename); | 
					
						
							|  |  |  | 		AST_APP_ARG(options); | 
					
						
							|  |  |  | 		AST_APP_ARG(post_process); | 
					
						
							|  |  |  | 	); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-10-26 19:48:14 +00:00
										 |  |  | 	if (ast_strlen_zero(data)) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "MixMonitor requires an argument (filename)\n"); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	u = ast_module_user_add(chan); | 
					
						
							| 
									
										
										
										
											2005-10-19 18:19:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-10 13:22:15 +00:00
										 |  |  | 	parse = ast_strdupa(data); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	AST_STANDARD_APP_ARGS(args, parse); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (ast_strlen_zero(args.filename)) { | 
					
						
							| 
									
										
										
										
											2005-11-04 22:11:51 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "MixMonitor requires an argument (filename)\n"); | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 		ast_module_user_remove(u); | 
					
						
							| 
									
										
										
										
											2005-10-19 18:19:02 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-13 16:02:02 +00:00
										 |  |  | 	if (args.options) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 		char *opts[OPT_ARG_ARRAY_SIZE] = { NULL, }; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 		ast_app_parse_options(mixmonitor_opts, &flags, opts, args.options); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (ast_test_flag(&flags, MUXFLAG_READVOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_READVOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the heard volume ('v') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_READVOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Heard volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_READVOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				readvol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		if (ast_test_flag(&flags, MUXFLAG_WRITEVOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_WRITEVOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the spoken volume ('V') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_WRITEVOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Spoken volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_WRITEVOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				writevol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		if (ast_test_flag(&flags, MUXFLAG_VOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_VOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the combined volume ('W') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_VOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Combined volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_VOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				readvol = writevol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* if not provided an absolute path, use the system-configured monitoring directory */ | 
					
						
							|  |  |  | 	if (args.filename[0] != '/') { | 
					
						
							|  |  |  | 		char *build; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		build = alloca(strlen(ast_config_AST_MONITOR_DIR) + strlen(args.filename) + 3); | 
					
						
							|  |  |  | 		sprintf(build, "%s/%s", ast_config_AST_MONITOR_DIR, args.filename); | 
					
						
							|  |  |  | 		args.filename = build; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbx_builtin_setvar_helper(chan, "MIXMONITOR_FILENAME", args.filename); | 
					
						
							| 
									
										
										
										
											2007-11-13 16:02:02 +00:00
										 |  |  | 	launch_monitor_thread(chan, args.filename, flags.flags, readvol, writevol, args.post_process); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	ast_module_user_remove(u); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | static int stop_mixmonitor_exec(struct ast_channel *chan, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	struct ast_module_user *u; | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	u = ast_module_user_add(chan); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 	ast_audiohook_detach_source(chan, mixmonitor_spy_type); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	ast_module_user_remove(u); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static int mixmonitor_cli(int fd, int argc, char **argv)  | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *chan; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (argc < 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!(chan = ast_get_channel_by_name_prefix_locked(argv[2], strlen(argv[2])))) { | 
					
						
							|  |  |  | 		ast_cli(fd, "No channel matching '%s' found.\n", argv[2]); | 
					
						
							|  |  |  | 		return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!strcasecmp(argv[1], "start")) | 
					
						
							|  |  |  | 		mixmonitor_exec(chan, argv[3]); | 
					
						
							| 
									
										
										
										
											2006-01-03 17:25:39 +00:00
										 |  |  | 	else if (!strcasecmp(argv[1], "stop")) | 
					
						
							| 
									
										
										
										
											2008-01-16 20:33:47 +00:00
										 |  |  | 		ast_audiohook_detach_source(chan, mixmonitor_spy_type); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-08 11:30:32 +00:00
										 |  |  | 	ast_channel_unlock(chan); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | static char *complete_mixmonitor_cli(const char *line, const char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ast_complete_channels(line, word, pos, state, 2); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | static struct ast_cli_entry cli_mixmonitor[] = { | 
					
						
							|  |  |  | 	{ { "mixmonitor", NULL, NULL }, | 
					
						
							|  |  |  | 	mixmonitor_cli, "Execute a MixMonitor command.", | 
					
						
							| 
									
										
										
										
											2006-02-17 04:17:35 +00:00
										 |  |  | 	"mixmonitor <start|stop> <chan_name> [args]\n\n" | 
					
						
							|  |  |  | 	"The optional arguments are passed to the\n" | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 	"MixMonitor application when the 'start' command is used.\n", | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	complete_mixmonitor_cli }, | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int unload_module(void) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_unregister_multiple(cli_mixmonitor, sizeof(cli_mixmonitor) / sizeof(struct ast_cli_entry)); | 
					
						
							|  |  |  | 	res = ast_unregister_application(stop_app); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	res |= ast_unregister_application(app); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	ast_module_user_hangup_all(); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int load_module(void) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_register_multiple(cli_mixmonitor, sizeof(cli_mixmonitor) / sizeof(struct ast_cli_entry)); | 
					
						
							|  |  |  | 	res = ast_register_application(app, mixmonitor_exec, synopsis, desc); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 	res |= ast_register_application(stop_app, stop_mixmonitor_exec, stop_synopsis, stop_desc); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Mixed Audio Monitoring Application"); |