| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-01-21 07:06:25 +00:00
										 |  |  |  * Copyright (C) 1999 - 2005, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * WaitForSilence Application by David C. Troy <dave@popvox.com> | 
					
						
							|  |  |  |  * Version 1.00 2004-01-29 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  |  * Mark Spencer <markster@digium.com> | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  |  * \brief Wait for Silence | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  *   - Waits for up to 'x' milliseconds of silence, 'y' times \n | 
					
						
							|  |  |  |  *   - WaitForSilence(500,2) will wait for 1/2 second of silence, twice \n | 
					
						
							|  |  |  |  *   - WaitForSilence(1000,1) will wait for 1 second of silence, once \n | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup applications | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:39:32 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2005-11-08 04:48:00 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:39:32 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2005-11-08 04:48:00 +00:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:39:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-21 06:02:45 +00:00
										 |  |  | #include "asterisk/file.h"
 | 
					
						
							|  |  |  | #include "asterisk/logger.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							|  |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/dsp.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/options.h"
 | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *tdesc = "Wait For Silence"; | 
					
						
							|  |  |  | static char *app = "WaitForSilence"; | 
					
						
							|  |  |  | static char *synopsis = "Waits for a specified amount of silence"; | 
					
						
							|  |  |  | static char *descrip =  | 
					
						
							|  |  |  | "  WaitForSilence(x[|y]) Wait for Silence: Waits for up to 'x' \n" | 
					
						
							|  |  |  | "milliseconds of silence, 'y' times or 1 if omitted\n" | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | "Set the channel variable WAITSTATUS with to one of these values:" | 
					
						
							|  |  |  | "SILENCE - if silence of x ms was detected" | 
					
						
							|  |  |  | "TIMEOUT - if silence of x ms was not detected." | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | "Examples:\n" | 
					
						
							| 
									
										
										
										
											2005-12-23 00:00:11 +00:00
										 |  |  | "  - WaitForSilence(500|2) will wait for 1/2 second of silence, twice\n" | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | "  - WaitForSilence(1000) will wait for 1 second of silence, once\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STANDARD_LOCAL_USER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LOCAL_USER_DECL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | static int do_waiting(struct ast_channel *chan, int maxsilence) { | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	struct ast_frame *f; | 
					
						
							|  |  |  | 	int totalsilence = 0; | 
					
						
							|  |  |  | 	int dspsilence = 0; | 
					
						
							|  |  |  | 	int gotsilence = 0;  | 
					
						
							| 
									
										
										
										
											2006-05-22 22:06:43 +00:00
										 |  |  | 	static int silencethreshold = 128; | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	int rfmt = 0; | 
					
						
							|  |  |  | 	int res = 0; | 
					
						
							|  |  |  | 	struct ast_dsp *sildet;	 /* silence detector dsp */ | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | 	time_t start, now; | 
					
						
							|  |  |  | 	time(&start); | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	rfmt = chan->readformat; /* Set to linear mode */ | 
					
						
							|  |  |  | 	res = ast_set_read_format(chan, AST_FORMAT_SLINEAR); | 
					
						
							|  |  |  | 	if (res < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to set to linear mode, giving up\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sildet = ast_dsp_new(); /* Create the silence detector */ | 
					
						
							|  |  |  | 	if (!sildet) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to create silence detector :(\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_dsp_set_threshold(sildet, silencethreshold); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Await silence... */ | 
					
						
							|  |  |  | 	f = NULL; | 
					
						
							|  |  |  | 	for(;;) { | 
					
						
							|  |  |  | 		res = ast_waitfor(chan, 2000); | 
					
						
							|  |  |  | 		if (!res) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "One waitfor failed, trying another\n"); | 
					
						
							|  |  |  | 			/* Try one more time in case of masq */ | 
					
						
							|  |  |  | 			res = ast_waitfor(chan, 2000); | 
					
						
							|  |  |  | 			if (!res) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "No audio available on %s??\n", chan->name); | 
					
						
							|  |  |  | 				res = -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (res < 0) { | 
					
						
							|  |  |  | 			f = NULL; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f = ast_read(chan); | 
					
						
							|  |  |  | 		if (!f) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		if (f->frametype == AST_FRAME_VOICE) { | 
					
						
							|  |  |  | 			dspsilence = 0; | 
					
						
							|  |  |  | 			ast_dsp_silence(sildet, f, &dspsilence); | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | 			if (dspsilence) { | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 				totalsilence = dspsilence; | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | 				time(&start); | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 				totalsilence = 0; | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 			if (totalsilence >= maxsilence) { | 
					
						
							|  |  |  | 				if (option_verbose > 2) | 
					
						
							|  |  |  | 					ast_verbose(VERBOSE_PREFIX_3 "Exiting with %dms silence > %dms required\n", totalsilence, maxsilence); | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 				/* Ended happily with silence */ | 
					
						
							|  |  |  | 				gotsilence = 1; | 
					
						
							| 
									
										
										
										
											2005-06-09 21:04:16 +00:00
										 |  |  | 				pbx_builtin_setvar_helper(chan, "WAITSTATUS", "SILENCE"); | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "WAITSTATUS was set to SILENCE\n"); | 
					
						
							|  |  |  | 				ast_frfree(f); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} else if ( difftime(time(&now),start) >= maxsilence/1000 ) { | 
					
						
							|  |  |  | 				pbx_builtin_setvar_helper(chan, "WAITSTATUS", "TIMEOUT"); | 
					
						
							|  |  |  | 				ast_log(LOG_DEBUG, "WAITSTATUS was set to TIMEOUT\n"); | 
					
						
							|  |  |  | 				ast_frfree(f); | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ast_frfree(f); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	if (rfmt && ast_set_read_format(chan, rfmt)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to restore format %s to channel '%s'\n", ast_getformatname(rfmt), chan->name); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	ast_dsp_free(sildet); | 
					
						
							|  |  |  | 	return gotsilence; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int waitforsilence_exec(struct ast_channel *chan, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	int res = 1; | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	struct localuser *u; | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	int maxsilence = 1000; | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	int iterations = 1, i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-19 18:19:02 +00:00
										 |  |  | 	LOCAL_USER_ADD(u); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	res = ast_answer(chan); /* Answer the channel */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	if (!data || ((sscanf(data, "%d|%d", &maxsilence, &iterations) != 2) && | 
					
						
							|  |  |  | 		(sscanf(data, "%d", &maxsilence) != 1))) { | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Using default value of 1000ms, 1 iteration\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	if (option_verbose > 2) | 
					
						
							|  |  |  | 		ast_verbose(VERBOSE_PREFIX_3 "Waiting %d time(s) for %d ms silence\n", iterations, maxsilence); | 
					
						
							| 
									
										
										
										
											2005-10-19 18:19:02 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	res = 1; | 
					
						
							| 
									
										
										
										
											2005-03-12 06:00:05 +00:00
										 |  |  | 	for (i=0; (i<iterations) && (res == 1); i++) { | 
					
						
							|  |  |  | 		res = do_waiting(chan, maxsilence); | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	LOCAL_USER_REMOVE(u); | 
					
						
							|  |  |  | 	if (res > 0) | 
					
						
							|  |  |  | 		res = 0; | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int unload_module(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = ast_unregister_application(app); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | 	STANDARD_HANGUP_LOCALUSERS; | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2004-11-24 05:01:49 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int load_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ast_register_application(app, waitforsilence_exec, synopsis, descrip); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *description(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return tdesc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int usecount(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	STANDARD_USECOUNT(res); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *key() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ASTERISK_GPL_KEY; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |