| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Copyright (C) 1999 - 2005, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +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-26 13:03:17 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-26 13:03:17 +00:00
										 |  |  |  * \brief Translate between signed linear and Speex (Open Codec) | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-26 13:03:17 +00:00
										 |  |  |  * http://www.speex.org
 | 
					
						
							|  |  |  |  * \note This work was motivated by Jeremy McNamara  | 
					
						
							|  |  |  |  * hacked to be configurable by anthm and bkw 9/28/2004 | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup codecs | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:12:19 +00:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <speex.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | /* We require a post 1.1.8 version of Speex to enable preprocessing
 | 
					
						
							|  |  |  |    and better type handling */    | 
					
						
							|  |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | #include <speex/speex_preprocess.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int quality = 3; | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | static int complexity = 2; | 
					
						
							|  |  |  | static int enhancement = 0; | 
					
						
							|  |  |  | static int vad = 0; | 
					
						
							|  |  |  | static int vbr = 0; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | static float vbr_quality = 4; | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | static int abr = 0; | 
					
						
							|  |  |  | static int dtx = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | static int preproc = 0; | 
					
						
							|  |  |  | static int pp_vad = 0; | 
					
						
							|  |  |  | static int pp_agc = 0; | 
					
						
							|  |  |  | static float pp_agc_level = 8000; | 
					
						
							|  |  |  | static int pp_denoise = 0; | 
					
						
							|  |  |  | static int pp_dereverb = 0; | 
					
						
							|  |  |  | static float pp_dereverb_decay = 0.4; | 
					
						
							|  |  |  | static float pp_dereverb_level = 0.3; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | #define TYPE_SILENCE	 0x2
 | 
					
						
							|  |  |  | #define TYPE_HIGH	 0x0
 | 
					
						
							|  |  |  | #define TYPE_LOW	 0x1
 | 
					
						
							|  |  |  | #define TYPE_MASK	 0x3
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:12:19 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 23:12:05 +00:00
										 |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							| 
									
										
										
										
											2005-06-06 22:12:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-21 06:02:45 +00:00
										 |  |  | #include "asterisk/lock.h"
 | 
					
						
							|  |  |  | #include "asterisk/translate.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/config.h"
 | 
					
						
							|  |  |  | #include "asterisk/options.h"
 | 
					
						
							|  |  |  | #include "asterisk/logger.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Sample frame data */ | 
					
						
							|  |  |  | #include "slin_speex_ex.h"
 | 
					
						
							|  |  |  | #include "speex_slin_ex.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-09 01:45:08 +00:00
										 |  |  | AST_MUTEX_DEFINE_STATIC(localuser_lock); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | static int localusecnt=0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *tdesc = "Speex/PCM16 (signed linear) Codec Translator"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_translator_pvt { | 
					
						
							|  |  |  | 	void *speex; | 
					
						
							|  |  |  | 	struct ast_frame f; | 
					
						
							|  |  |  | 	SpeexBits bits; | 
					
						
							|  |  |  | 	int framesize; | 
					
						
							|  |  |  | 	/* Space to build offset */ | 
					
						
							|  |  |  | 	char offset[AST_FRIENDLY_OFFSET]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 	SpeexPreprocessState *pp; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* Buffer for our outgoing frame */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	spx_int16_t outbuf[8000]; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* Enough to store a full second */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	spx_int16_t buf[8000]; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	short outbuf[8000]; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	short buf[8000]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	int tail; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	int silent_state; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define speex_coder_pvt ast_translator_pvt
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-30 22:55:42 +00:00
										 |  |  | static struct ast_translator_pvt *lintospeex_new(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct speex_coder_pvt *tmp; | 
					
						
							|  |  |  | 	tmp = malloc(sizeof(struct speex_coder_pvt)); | 
					
						
							|  |  |  | 	if (tmp) { | 
					
						
							|  |  |  | 		if (!(tmp->speex = speex_encoder_init(&speex_nb_mode))) { | 
					
						
							|  |  |  | 			free(tmp); | 
					
						
							|  |  |  | 			tmp = NULL; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			speex_bits_init(&tmp->bits); | 
					
						
							|  |  |  | 			speex_bits_reset(&tmp->bits); | 
					
						
							|  |  |  | 			speex_encoder_ctl(tmp->speex, SPEEX_GET_FRAME_SIZE, &tmp->framesize); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 			speex_encoder_ctl(tmp->speex, SPEEX_SET_COMPLEXITY, &complexity); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 			if (preproc) { | 
					
						
							|  |  |  | 				tmp->pp = speex_preprocess_state_init(tmp->framesize, 8000); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_VAD, &pp_vad); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_AGC, &pp_agc); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_AGC_LEVEL, &pp_agc_level); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_DENOISE, &pp_denoise); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_DEREVERB, &pp_dereverb); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_DEREVERB_DECAY, &pp_dereverb_decay); | 
					
						
							|  |  |  | 				speex_preprocess_ctl(tmp->pp, SPEEX_PREPROCESS_SET_DEREVERB_LEVEL, &pp_dereverb_level); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			if (!abr && !vbr) { | 
					
						
							| 
									
										
										
										
											2004-12-30 01:40:47 +00:00
										 |  |  | 				speex_encoder_ctl(tmp->speex, SPEEX_SET_QUALITY, &quality); | 
					
						
							|  |  |  | 				if (vad) | 
					
						
							|  |  |  | 					speex_encoder_ctl(tmp->speex, SPEEX_SET_VAD, &vad); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 			if (vbr) { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				speex_encoder_ctl(tmp->speex, SPEEX_SET_VBR, &vbr); | 
					
						
							|  |  |  | 				speex_encoder_ctl(tmp->speex, SPEEX_SET_VBR_QUALITY, &vbr_quality); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 			if (abr) { | 
					
						
							| 
									
										
										
										
											2004-12-30 01:40:47 +00:00
										 |  |  | 				speex_encoder_ctl(tmp->speex, SPEEX_SET_ABR, &abr); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 			if (dtx) | 
					
						
							|  |  |  | 				speex_encoder_ctl(tmp->speex, SPEEX_SET_DTX, &dtx);  | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 			tmp->tail = 0; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 			tmp->silent_state = 0; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		localusecnt++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return tmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-30 22:55:42 +00:00
										 |  |  | static struct ast_translator_pvt *speextolin_new(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct speex_coder_pvt *tmp; | 
					
						
							|  |  |  | 	tmp = malloc(sizeof(struct speex_coder_pvt)); | 
					
						
							|  |  |  | 	if (tmp) { | 
					
						
							|  |  |  | 		if (!(tmp->speex = speex_decoder_init(&speex_nb_mode))) { | 
					
						
							|  |  |  | 			free(tmp); | 
					
						
							|  |  |  | 			tmp = NULL; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			speex_bits_init(&tmp->bits); | 
					
						
							|  |  |  | 			speex_decoder_ctl(tmp->speex, SPEEX_GET_FRAME_SIZE, &tmp->framesize); | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 			if (enhancement) | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				speex_decoder_ctl(tmp->speex, SPEEX_SET_ENH, &enhancement); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 			tmp->tail = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		localusecnt++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return tmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-30 22:55:42 +00:00
										 |  |  | static struct ast_frame *lintospeex_sample(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	static struct ast_frame f; | 
					
						
							|  |  |  | 	f.frametype = AST_FRAME_VOICE; | 
					
						
							|  |  |  | 	f.subclass = AST_FORMAT_SLINEAR; | 
					
						
							|  |  |  | 	f.datalen = sizeof(slin_speex_ex); | 
					
						
							|  |  |  | 	/* Assume 8000 Hz */ | 
					
						
							| 
									
										
										
										
											2003-01-28 06:57:51 +00:00
										 |  |  | 	f.samples = sizeof(slin_speex_ex)/2; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	f.mallocd = 0; | 
					
						
							|  |  |  | 	f.offset = 0; | 
					
						
							|  |  |  | 	f.src = __PRETTY_FUNCTION__; | 
					
						
							|  |  |  | 	f.data = slin_speex_ex; | 
					
						
							|  |  |  | 	return &f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-30 22:55:42 +00:00
										 |  |  | static struct ast_frame *speextolin_sample(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	static struct ast_frame f; | 
					
						
							|  |  |  | 	f.frametype = AST_FRAME_VOICE; | 
					
						
							|  |  |  | 	f.subclass = AST_FORMAT_SPEEX; | 
					
						
							|  |  |  | 	f.datalen = sizeof(speex_slin_ex); | 
					
						
							|  |  |  | 	/* All frames are 20 ms long */ | 
					
						
							| 
									
										
										
										
											2003-01-28 06:57:51 +00:00
										 |  |  | 	f.samples = 160; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	f.mallocd = 0; | 
					
						
							|  |  |  | 	f.offset = 0; | 
					
						
							|  |  |  | 	f.src = __PRETTY_FUNCTION__; | 
					
						
							|  |  |  | 	f.data = speex_slin_ex; | 
					
						
							|  |  |  | 	return &f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_frame *speextolin_frameout(struct ast_translator_pvt *tmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!tmp->tail) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	/* Signed linear is no particular frame size, so just send whatever
 | 
					
						
							|  |  |  | 	   we have in the buffer in one lump sum */ | 
					
						
							|  |  |  | 	tmp->f.frametype = AST_FRAME_VOICE; | 
					
						
							|  |  |  | 	tmp->f.subclass = AST_FORMAT_SLINEAR; | 
					
						
							|  |  |  | 	tmp->f.datalen = tmp->tail * 2; | 
					
						
							|  |  |  | 	/* Assume 8000 Hz */ | 
					
						
							| 
									
										
										
										
											2003-01-28 06:57:51 +00:00
										 |  |  | 	tmp->f.samples = tmp->tail; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	tmp->f.mallocd = 0; | 
					
						
							|  |  |  | 	tmp->f.offset = AST_FRIENDLY_OFFSET; | 
					
						
							|  |  |  | 	tmp->f.src = __PRETTY_FUNCTION__; | 
					
						
							|  |  |  | 	tmp->f.data = tmp->buf; | 
					
						
							|  |  |  | 	/* Reset tail pointer */ | 
					
						
							|  |  |  | 	tmp->tail = 0; | 
					
						
							|  |  |  | 	return &tmp->f;	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int speextolin_framein(struct ast_translator_pvt *tmp, struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Assuming there's space left, decode into the current buffer at
 | 
					
						
							|  |  |  | 	   the tail location.  Read in as many frames as there are */ | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 	spx_int16_t out[1024]; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	float fout[1024]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	if (f->datalen == 0) {  /* Native PLC interpolation */ | 
					
						
							|  |  |  | 		if (tmp->tail + tmp->framesize > sizeof(tmp->buf) / 2) { | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Out of buffer space\n"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 		speex_decode_int(tmp->speex, NULL, tmp->buf + tmp->tail); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 		speex_decode(tmp->speex, NULL, fout); | 
					
						
							|  |  |  | 		for (x=0;x<tmp->framesize;x++) { | 
					
						
							|  |  |  | 			tmp->buf[tmp->tail + x] = fout[x]; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 		tmp->tail += tmp->framesize; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* Read in bits */ | 
					
						
							|  |  |  | 	speex_bits_read_from(&tmp->bits, f->data, f->datalen); | 
					
						
							|  |  |  | 	for(;;) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 		res = speex_decode_int(tmp->speex, &tmp->bits, out); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		res = speex_decode(tmp->speex, &tmp->bits, fout); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		if (res < 0) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		if (tmp->tail + tmp->framesize < sizeof(tmp->buf) / 2) { | 
					
						
							|  |  |  | 			for (x=0;x<tmp->framesize;x++) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 				tmp->buf[tmp->tail + x] = out[x]; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 				tmp->buf[tmp->tail + x] = fout[x]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			tmp->tail += tmp->framesize; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Out of buffer space\n"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lintospeex_framein(struct ast_translator_pvt *tmp, struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Just add the frames to our stream */ | 
					
						
							|  |  |  | 	/* XXX We should look at how old the rest of our stream is, and if it
 | 
					
						
							|  |  |  | 	   is too old, then we should overwrite it entirely, otherwise we can | 
					
						
							|  |  |  | 	   get artifacts of earlier talk that do not belong */ | 
					
						
							|  |  |  | 	if (tmp->tail + f->datalen/2 < sizeof(tmp->buf) / 2) { | 
					
						
							|  |  |  | 		memcpy((tmp->buf + tmp->tail), f->data, f->datalen); | 
					
						
							|  |  |  | 		tmp->tail += f->datalen/2; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Out of buffer space\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_frame *lintospeex_frameout(struct ast_translator_pvt *tmp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifndef _SPEEX_TYPES_H
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	float fbuf[1024]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	int x; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	int len; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	int y=0; | 
					
						
							|  |  |  | 	int is_speech=1; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* We can't work on anything less than a frame in size */ | 
					
						
							|  |  |  | 	if (tmp->tail < tmp->framesize) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	tmp->f.frametype = AST_FRAME_VOICE; | 
					
						
							|  |  |  | 	tmp->f.subclass = AST_FORMAT_SPEEX; | 
					
						
							|  |  |  | 	tmp->f.mallocd = 0; | 
					
						
							|  |  |  | 	tmp->f.offset = AST_FRIENDLY_OFFSET; | 
					
						
							|  |  |  | 	tmp->f.src = __PRETTY_FUNCTION__; | 
					
						
							|  |  |  | 	tmp->f.data = tmp->outbuf; | 
					
						
							|  |  |  | 	speex_bits_reset(&tmp->bits); | 
					
						
							|  |  |  | 	while(tmp->tail >= tmp->framesize) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 		/* Preprocess audio */ | 
					
						
							|  |  |  | 		if(preproc) | 
					
						
							|  |  |  | 			is_speech = speex_preprocess(tmp->pp, tmp->buf, NULL); | 
					
						
							|  |  |  | 		/* Encode a frame of data */ | 
					
						
							|  |  |  | 		if (is_speech) { | 
					
						
							|  |  |  | 			/* If DTX enabled speex_encode returns 0 during silence */ | 
					
						
							|  |  |  | 			is_speech = speex_encode_int(tmp->speex, tmp->buf, &tmp->bits) || !dtx; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* 5 zeros interpreted by Speex as silence (submode 0) */ | 
					
						
							|  |  |  | 			speex_bits_pack(&tmp->bits, 0, 5); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		/* Convert to floating point */ | 
					
						
							|  |  |  | 		for (x=0;x<tmp->framesize;x++) | 
					
						
							|  |  |  | 			fbuf[x] = tmp->buf[x]; | 
					
						
							|  |  |  | 		/* Encode a frame of data */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		is_speech = speex_encode(tmp->speex, fbuf, &tmp->bits) || !dtx; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		/* Assume 8000 Hz -- 20 ms */ | 
					
						
							|  |  |  | 		tmp->tail -= tmp->framesize; | 
					
						
							|  |  |  | 		/* Move the data at the end of the buffer to the front */ | 
					
						
							|  |  |  | 		if (tmp->tail) | 
					
						
							|  |  |  | 			memmove(tmp->buf, tmp->buf + tmp->framesize, tmp->tail * 2); | 
					
						
							|  |  |  | 		y++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-12 22:20:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	/* Use AST_FRAME_CNG to signify the start of any silence period */ | 
					
						
							|  |  |  | 	if (!is_speech) { | 
					
						
							|  |  |  | 		if (tmp->silent_state) { | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			tmp->silent_state = 1; | 
					
						
							|  |  |  | 			speex_bits_reset(&tmp->bits); | 
					
						
							|  |  |  | 			tmp->f.frametype = AST_FRAME_CNG; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		tmp->silent_state = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-12 22:20:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* Terminate bit stream */ | 
					
						
							|  |  |  | 	speex_bits_pack(&tmp->bits, 15, 5); | 
					
						
							|  |  |  | 	len = speex_bits_write(&tmp->bits, (char *)tmp->outbuf, sizeof(tmp->outbuf)); | 
					
						
							|  |  |  | 	tmp->f.datalen = len; | 
					
						
							| 
									
										
										
										
											2003-01-28 06:57:51 +00:00
										 |  |  | 	tmp->f.samples = y * 160; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		static int fd = -1; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		if (fd < 0) { | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 			fd = open("speex.raw", O_WRONLY|O_TRUNC|O_CREAT); | 
					
						
							|  |  |  | 			if (fd > -1) { | 
					
						
							|  |  |  | 				write(fd, tmp->f.data, tmp->f.datalen); | 
					
						
							|  |  |  | 				close(fd); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	return &tmp->f;	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void speextolin_destroy(struct ast_translator_pvt *pvt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	speex_decoder_destroy(pvt->speex); | 
					
						
							|  |  |  | 	speex_bits_destroy(&pvt->bits); | 
					
						
							|  |  |  | 	free(pvt); | 
					
						
							|  |  |  | 	localusecnt--; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void lintospeex_destroy(struct ast_translator_pvt *pvt) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 	if (preproc) | 
					
						
							|  |  |  | 		speex_preprocess_state_destroy(pvt->pp); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	speex_encoder_destroy(pvt->speex); | 
					
						
							|  |  |  | 	speex_bits_destroy(&pvt->bits); | 
					
						
							|  |  |  | 	free(pvt); | 
					
						
							|  |  |  | 	localusecnt--; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_translator speextolin = | 
					
						
							|  |  |  | 	{ "speextolin",  | 
					
						
							|  |  |  | 	   AST_FORMAT_SPEEX, AST_FORMAT_SLINEAR, | 
					
						
							|  |  |  | 	   speextolin_new, | 
					
						
							|  |  |  | 	   speextolin_framein, | 
					
						
							|  |  |  | 	   speextolin_frameout, | 
					
						
							|  |  |  | 	   speextolin_destroy, | 
					
						
							|  |  |  | 	   speextolin_sample | 
					
						
							|  |  |  | 	   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_translator lintospeex = | 
					
						
							|  |  |  | 	{ "lintospeex",  | 
					
						
							|  |  |  | 	   AST_FORMAT_SLINEAR, AST_FORMAT_SPEEX, | 
					
						
							|  |  |  | 	   lintospeex_new, | 
					
						
							|  |  |  | 	   lintospeex_framein, | 
					
						
							|  |  |  | 	   lintospeex_frameout, | 
					
						
							|  |  |  | 	   lintospeex_destroy, | 
					
						
							|  |  |  | 	   lintospeex_sample | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | static void parse_config(void)  | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	struct ast_config *cfg; | 
					
						
							|  |  |  | 	struct ast_variable *var; | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2005-06-21 23:16:32 +00:00
										 |  |  | 	float res_f; | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-25 06:10:20 +00:00
										 |  |  | 	if ((cfg = ast_config_load("codecs.conf"))) { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 		if ((var = ast_variable_browse(cfg, "speex"))) { | 
					
						
							|  |  |  | 			while (var) { | 
					
						
							|  |  |  | 				if (!strcasecmp(var->name, "quality")) { | 
					
						
							|  |  |  | 					res = abs(atoi(var->value)); | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 					if (res > -1 && res < 11) { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting Quality to %d\n",res); | 
					
						
							|  |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						quality = res; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 					} else  | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error Quality must be 0-10\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "complexity")) { | 
					
						
							|  |  |  | 					res = abs(atoi(var->value)); | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 					if (res > -1 && res < 11) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting Complexity to %d\n",res); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						complexity = res; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 					} else  | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error! Complexity must be 0-10\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "vbr_quality")) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 					if (sscanf(var->value, "%f", &res_f) == 1 && res_f >= 0 && res_f <= 10) { | 
					
						
							|  |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting VBR Quality to %f\n",res_f); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2005-06-21 23:16:32 +00:00
										 |  |  | 						vbr_quality = res_f; | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 					} else | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | 						ast_log(LOG_ERROR,"Error! VBR Quality must be 0-10\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "abr_quality")) { | 
					
						
							| 
									
										
										
										
											2004-12-30 01:40:47 +00:00
										 |  |  | 					ast_log(LOG_ERROR,"Error! ABR Quality setting obsolete, set ABR to desired bitrate\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "enhancement")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					enhancement = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Perceptual Enhancement Mode. [%s]\n",enhancement ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "vbr")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					vbr = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: VBR Mode. [%s]\n",vbr ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "abr")) { | 
					
						
							| 
									
										
										
										
											2004-12-30 01:40:47 +00:00
										 |  |  | 					res = abs(atoi(var->value)); | 
					
						
							|  |  |  | 					if (res >= 0) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 						if (option_verbose > 2) { | 
					
						
							|  |  |  | 							if (res > 0) | 
					
						
							|  |  |  | 								ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting ABR target bitrate to %d\n",res); | 
					
						
							|  |  |  | 							else | 
					
						
							|  |  |  | 								ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Disabling ABR\n"); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2004-12-30 01:40:47 +00:00
										 |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						abr = res; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 					} else  | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error! ABR target bitrate must be >= 0\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "vad")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					vad = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							| 
									
										
										
										
											2005-06-21 23:16:32 +00:00
										 |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: VAD Mode. [%s]\n",vad ? "on" : "off"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "dtx")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					dtx = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: DTX Mode. [%s]\n",dtx ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 				} else if (!strcasecmp(var->name, "preprocess")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					preproc = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessing. [%s]\n",preproc ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_vad")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					pp_vad = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessor VAD. [%s]\n",pp_vad ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_agc")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					pp_agc = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessor AGC. [%s]\n",pp_agc ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_agc_level")) { | 
					
						
							|  |  |  | 					if (sscanf(var->value, "%f", &res_f) == 1 && res_f >= 0) { | 
					
						
							|  |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting preprocessor AGC Level to %f\n",res_f); | 
					
						
							|  |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						pp_agc_level = res_f; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 					} else | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error! Preprocessor AGC Level must be >= 0\n"); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_denoise")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					pp_denoise = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessor Denoise. [%s]\n",pp_denoise ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_dereverb")) { | 
					
						
							|  |  |  | 					ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 					pp_dereverb = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 					if (option_verbose > 2) | 
					
						
							|  |  |  | 						ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessor Dereverb. [%s]\n",pp_dereverb ? "on" : "off"); | 
					
						
							|  |  |  | 					ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_dereverb_decay")) { | 
					
						
							|  |  |  | 					if (sscanf(var->value, "%f", &res_f) == 1 && res_f >= 0) { | 
					
						
							|  |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting preprocessor Dereverb Decay to %f\n",res_f); | 
					
						
							|  |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						pp_dereverb_decay = res_f; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 					} else | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error! Preprocessor Dereverb Decay must be >= 0\n"); | 
					
						
							|  |  |  | 				} else if (!strcasecmp(var->name, "pp_dereverb_level")) { | 
					
						
							|  |  |  | 					if (sscanf(var->value, "%f", &res_f) == 1 && res_f >= 0) { | 
					
						
							|  |  |  | 						if (option_verbose > 2) | 
					
						
							|  |  |  | 							ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting preprocessor Dereverb Level to %f\n",res_f); | 
					
						
							|  |  |  | 						ast_mutex_lock(&localuser_lock); | 
					
						
							|  |  |  | 						pp_dereverb_level = res_f; | 
					
						
							|  |  |  | 						ast_mutex_unlock(&localuser_lock); | 
					
						
							|  |  |  | 					} else | 
					
						
							|  |  |  | 						ast_log(LOG_ERROR,"Error! Preprocessor Dereverb Level must be >= 0\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				var = var->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-05-12 17:40:44 +00:00
										 |  |  | 		ast_config_destroy(cfg); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | int reload(void)  | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	parse_config(); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int unload_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	res = ast_unregister_translator(&lintospeex); | 
					
						
							|  |  |  | 	if (!res) | 
					
						
							|  |  |  | 		res = ast_unregister_translator(&speextolin); | 
					
						
							|  |  |  | 	if (localusecnt) | 
					
						
							|  |  |  | 		res = -1; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&localuser_lock); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int load_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	parse_config(); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	res=ast_register_translator(&speextolin); | 
					
						
							|  |  |  | 	if (!res)  | 
					
						
							|  |  |  | 		res=ast_register_translator(&lintospeex); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		ast_unregister_translator(&speextolin); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *description(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return tdesc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int usecount(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	STANDARD_USECOUNT(res); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *key() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ASTERISK_GPL_KEY; | 
					
						
							|  |  |  | } |