| 
									
										
										
										
											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
										 |  |  |  * \note This work was motivated by Jeremy McNamara  | 
					
						
							|  |  |  |  * hacked to be configurable by anthm and bkw 9/28/2004 | 
					
						
							| 
									
										
										
										
											2007-02-24 19:27:50 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup codecs | 
					
						
							| 
									
										
										
										
											2007-02-24 19:27:50 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \extref The Speex library - http://www.speex.org
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  | /*** MODULEINFO
 | 
					
						
							| 
									
										
										
										
											2006-07-05 20:12:29 +00:00
										 |  |  | 	<depend>speex</depend> | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  |  ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  | #include <speex/speex.h>
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:12:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +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"
 | 
					
						
							|  |  |  | #include "asterisk/utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Sample frame data */ | 
					
						
							|  |  |  | #include "slin_speex_ex.h"
 | 
					
						
							|  |  |  | #include "speex_slin_ex.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | /* codec variables */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 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; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | static int dtx = 0;	/* set to 1 to enable silence detection */ | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | static int preproc = 0; | 
					
						
							|  |  |  | static int pp_vad = 0; | 
					
						
							|  |  |  | static int pp_agc = 0; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | static float pp_agc_level = 8000; /* XXX what is this 8000 ? */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | #define	BUFFER_SAMPLES	8000
 | 
					
						
							|  |  |  | #define	SPEEX_SAMPLES	160
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | struct speex_coder_pvt { | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	void *speex; | 
					
						
							|  |  |  | 	SpeexBits bits; | 
					
						
							|  |  |  | 	int framesize; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	int silent_state; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 	SpeexPreprocessState *pp; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	spx_int16_t buf[BUFFER_SAMPLES]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	int16_t buf[BUFFER_SAMPLES];	/* input, waiting to be compressed */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | static int lintospeex_new(struct ast_trans_pvt *pvt) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *tmp = pvt->pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(tmp->speex = speex_encoder_init(&speex_nb_mode))) | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	speex_bits_init(&tmp->bits); | 
					
						
							|  |  |  | 	speex_bits_reset(&tmp->bits); | 
					
						
							|  |  |  | 	speex_encoder_ctl(tmp->speex, SPEEX_GET_FRAME_SIZE, &tmp->framesize); | 
					
						
							|  |  |  | 	speex_encoder_ctl(tmp->speex, SPEEX_SET_COMPLEXITY, &complexity); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (preproc) { | 
					
						
							|  |  |  | 		tmp->pp = speex_preprocess_state_init(tmp->framesize, 8000); /* XXX what is this 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); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (!abr && !vbr) { | 
					
						
							|  |  |  | 		speex_encoder_ctl(tmp->speex, SPEEX_SET_QUALITY, &quality); | 
					
						
							|  |  |  | 		if (vad) | 
					
						
							|  |  |  | 			speex_encoder_ctl(tmp->speex, SPEEX_SET_VAD, &vad); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (vbr) { | 
					
						
							|  |  |  | 		speex_encoder_ctl(tmp->speex, SPEEX_SET_VBR, &vbr); | 
					
						
							|  |  |  | 		speex_encoder_ctl(tmp->speex, SPEEX_SET_VBR_QUALITY, &vbr_quality); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (abr) | 
					
						
							|  |  |  | 		speex_encoder_ctl(tmp->speex, SPEEX_SET_ABR, &abr); | 
					
						
							|  |  |  | 	if (dtx) | 
					
						
							|  |  |  | 		speex_encoder_ctl(tmp->speex, SPEEX_SET_DTX, &dtx);  | 
					
						
							|  |  |  | 	tmp->silent_state = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | static int speextolin_new(struct ast_trans_pvt *pvt) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *tmp = pvt->pvt; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!(tmp->speex = speex_decoder_init(&speex_nb_mode))) | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	speex_bits_init(&tmp->bits); | 
					
						
							|  |  |  | 	speex_decoder_ctl(tmp->speex, SPEEX_GET_FRAME_SIZE, &tmp->framesize); | 
					
						
							|  |  |  | 	if (enhancement) | 
					
						
							|  |  |  | 		speex_decoder_ctl(tmp->speex, SPEEX_SET_ENH, &enhancement); | 
					
						
							| 
									
										
										
										
											2006-06-24 19:43:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	f.samples = SPEEX_SAMPLES; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | /*! \brief convert and store into outbuf */ | 
					
						
							|  |  |  | static int speextolin_framein(struct ast_trans_pvt *pvt, struct ast_frame *f) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *tmp = pvt->pvt; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* 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; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	int16_t *dst = (int16_t *)pvt->outbuf; | 
					
						
							|  |  |  | 	/* XXX fout is a temporary buffer, may have different types */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	spx_int16_t fout[1024]; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #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 */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		if (pvt->samples + tmp->framesize > BUFFER_SAMPLES) { | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		speex_decode_int(tmp->speex, NULL, dst + pvt->samples); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 		speex_decode(tmp->speex, NULL, fout); | 
					
						
							|  |  |  | 		for (x=0;x<tmp->framesize;x++) { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 			dst[pvt->samples + x] = (int16_t)fout[x]; | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		pvt->samples += tmp->framesize; | 
					
						
							| 
									
										
										
										
											2005-03-17 21:30:19 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* Read in bits */ | 
					
						
							|  |  |  | 	speex_bits_read_from(&tmp->bits, f->data, f->datalen); | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	for (;;) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		res = speex_decode_int(tmp->speex, &tmp->bits, fout); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #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; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		if (pvt->samples + tmp->framesize > BUFFER_SAMPLES) { | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 			ast_log(LOG_WARNING, "Out of buffer space\n"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		for (x = 0 ; x < tmp->framesize; x++) | 
					
						
							|  |  |  | 			dst[pvt->samples + x] = (int16_t)fout[x]; | 
					
						
							|  |  |  | 		pvt->samples += tmp->framesize; | 
					
						
							|  |  |  | 		pvt->datalen += 2 * tmp->framesize; /* 2 bytes/sample */ | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | /*! \brief store input frame in work buffer */ | 
					
						
							|  |  |  | static int lintospeex_framein(struct ast_trans_pvt *pvt, struct ast_frame *f) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *tmp = pvt->pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* 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 */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	memcpy(tmp->buf + pvt->samples, f->data, f->datalen); | 
					
						
							|  |  |  | 	pvt->samples += f->samples; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | /*! \brief convert work buffer and produce output frame */ | 
					
						
							|  |  |  | static struct ast_frame *lintospeex_frameout(struct ast_trans_pvt *pvt) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *tmp = pvt->pvt; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	int is_speech=1; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	int datalen = 0;	/* output bytes */ | 
					
						
							|  |  |  | 	int samples = 0;	/* output samples */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	/* We can't work on anything less than a frame in size */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (pvt->samples < tmp->framesize) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	speex_bits_reset(&tmp->bits); | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	while (pvt->samples >= tmp->framesize) { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #ifdef _SPEEX_TYPES_H
 | 
					
						
							|  |  |  | 		/* Preprocess audio */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		if (preproc) | 
					
						
							| 
									
										
										
										
											2006-09-08 21:33:32 +00:00
										 |  |  | 			is_speech = speex_preprocess(tmp->pp, tmp->buf + samples, NULL); | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		/* Encode a frame of data */ | 
					
						
							|  |  |  | 		if (is_speech) { | 
					
						
							|  |  |  | 			/* If DTX enabled speex_encode returns 0 during silence */ | 
					
						
							| 
									
										
										
										
											2006-09-08 21:33:32 +00:00
										 |  |  | 			is_speech = speex_encode_int(tmp->speex, tmp->buf + samples, &tmp->bits) || !dtx; | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			/* 5 zeros interpreted by Speex as silence (submode 0) */ | 
					
						
							|  |  |  | 			speex_bits_pack(&tmp->bits, 0, 5); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			float fbuf[1024]; | 
					
						
							|  |  |  | 			int x; | 
					
						
							|  |  |  | 			/* Convert to floating point */ | 
					
						
							|  |  |  | 			for (x = 0; x < tmp->framesize; x++) | 
					
						
							| 
									
										
										
										
											2006-09-08 21:33:32 +00:00
										 |  |  | 				fbuf[x] = tmp->buf[samples + x]; | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 			/* Encode a frame of data */ | 
					
						
							|  |  |  | 			is_speech = speex_encode(tmp->speex, fbuf, &tmp->bits) || !dtx; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 		samples += tmp->framesize; | 
					
						
							|  |  |  | 		pvt->samples -= tmp->framesize; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-12 22:20:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-08 21:33:32 +00:00
										 |  |  | 	/* Move the data at the end of the buffer to the front */ | 
					
						
							|  |  |  | 	if (pvt->samples) | 
					
						
							|  |  |  | 		memmove(tmp->buf, tmp->buf + samples, pvt->samples * 2); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 	/* Use AST_FRAME_CNG to signify the start of any silence period */ | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (is_speech) { | 
					
						
							|  |  |  | 		tmp->silent_state = 0; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		if (tmp->silent_state) { | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			tmp->silent_state = 1; | 
					
						
							|  |  |  | 			speex_bits_reset(&tmp->bits); | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 			memset(&pvt->f, 0, sizeof(pvt->f)); | 
					
						
							|  |  |  | 			pvt->f.frametype = AST_FRAME_CNG; | 
					
						
							|  |  |  | 			pvt->f.samples = samples; | 
					
						
							|  |  |  | 			/* XXX what now ? format etc... */ | 
					
						
							| 
									
										
										
										
											2005-08-26 20:14:06 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	datalen = speex_bits_write(&tmp->bits, pvt->outbuf, pvt->t->buf_size); | 
					
						
							|  |  |  | 	return ast_trans_frameout(pvt, datalen, samples); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | static void speextolin_destroy(struct ast_trans_pvt *arg) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *pvt = arg->pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	speex_decoder_destroy(pvt->speex); | 
					
						
							|  |  |  | 	speex_bits_destroy(&pvt->bits); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | static void lintospeex_destroy(struct ast_trans_pvt *arg) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct speex_coder_pvt *pvt = arg->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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | static struct ast_translator speextolin = { | 
					
						
							|  |  |  | 	.name = "speextolin",  | 
					
						
							|  |  |  | 	.srcfmt = AST_FORMAT_SPEEX, | 
					
						
							|  |  |  | 	.dstfmt =  AST_FORMAT_SLINEAR, | 
					
						
							|  |  |  | 	.newpvt = speextolin_new, | 
					
						
							|  |  |  | 	.framein = speextolin_framein, | 
					
						
							|  |  |  | 	.destroy = speextolin_destroy, | 
					
						
							|  |  |  | 	.sample = speextolin_sample, | 
					
						
							|  |  |  | 	.desc_size = sizeof(struct speex_coder_pvt), | 
					
						
							|  |  |  | 	.buffer_samples = BUFFER_SAMPLES, | 
					
						
							|  |  |  | 	.buf_size = BUFFER_SAMPLES * 2, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_translator lintospeex = { | 
					
						
							|  |  |  | 	.name = "lintospeex",  | 
					
						
							|  |  |  | 	.srcfmt = AST_FORMAT_SLINEAR, | 
					
						
							|  |  |  | 	.dstfmt = AST_FORMAT_SPEEX, | 
					
						
							|  |  |  | 	.newpvt = lintospeex_new, | 
					
						
							|  |  |  | 	.framein = lintospeex_framein, | 
					
						
							|  |  |  | 	.frameout = lintospeex_frameout, | 
					
						
							|  |  |  | 	.destroy = lintospeex_destroy, | 
					
						
							|  |  |  | 	.sample = lintospeex_sample, | 
					
						
							|  |  |  | 	.desc_size = sizeof(struct speex_coder_pvt), | 
					
						
							|  |  |  | 	.buffer_samples = BUFFER_SAMPLES, | 
					
						
							|  |  |  | 	.buf_size = BUFFER_SAMPLES * 2, /* XXX maybe a lot less ? */ | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | static void parse_config(void)  | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	struct ast_config *cfg = ast_config_load("codecs.conf"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	if (cfg == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (var = ast_variable_browse(cfg, "speex"); var; var = var->next) { | 
					
						
							|  |  |  | 		if (!strcasecmp(var->name, "quality")) { | 
					
						
							|  |  |  | 			res = abs(atoi(var->value)); | 
					
						
							|  |  |  | 			if (res > -1 && res < 11) { | 
					
						
							|  |  |  | 				if (option_verbose > 2) | 
					
						
							|  |  |  | 					ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting Quality to %d\n",res); | 
					
						
							|  |  |  | 				quality = res; | 
					
						
							|  |  |  | 			} else  | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error Quality must be 0-10\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "complexity")) { | 
					
						
							|  |  |  | 			res = abs(atoi(var->value)); | 
					
						
							|  |  |  | 			if (res > -1 && res < 11) { | 
					
						
							|  |  |  | 				if (option_verbose > 2) | 
					
						
							|  |  |  | 					ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Setting Complexity to %d\n",res); | 
					
						
							|  |  |  | 				complexity = res; | 
					
						
							|  |  |  | 			} else  | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error! Complexity must be 0-10\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "vbr_quality")) { | 
					
						
							|  |  |  | 			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); | 
					
						
							|  |  |  | 				vbr_quality = res_f; | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error! VBR Quality must be 0-10\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "abr_quality")) { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR,"Error! ABR Quality setting obsolete, set ABR to desired bitrate\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "enhancement")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "vbr")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "abr")) { | 
					
						
							|  |  |  | 			res = abs(atoi(var->value)); | 
					
						
							|  |  |  | 			if (res >= 0) { | 
					
						
							|  |  |  | 				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-10-01 03:18:48 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 				abr = res; | 
					
						
							|  |  |  | 			} else  | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error! ABR target bitrate must be >= 0\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "vad")) { | 
					
						
							|  |  |  | 			vad = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 			if (option_verbose > 2) | 
					
						
							|  |  |  | 				ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: VAD Mode. [%s]\n",vad ? "on" : "off"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "dtx")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "preprocess")) { | 
					
						
							|  |  |  | 			preproc = ast_true(var->value) ? 1 : 0; | 
					
						
							|  |  |  | 			if (option_verbose > 2) | 
					
						
							|  |  |  | 				ast_verbose(VERBOSE_PREFIX_3 "CODEC SPEEX: Preprocessing. [%s]\n",preproc ? "on" : "off"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "pp_vad")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "pp_agc")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} 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); | 
					
						
							|  |  |  | 				pp_agc_level = res_f; | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error! Preprocessor AGC Level must be >= 0\n"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "pp_denoise")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} else if (!strcasecmp(var->name, "pp_dereverb")) { | 
					
						
							|  |  |  | 			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"); | 
					
						
							|  |  |  | 		} 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); | 
					
						
							|  |  |  | 				pp_dereverb_decay = res_f; | 
					
						
							|  |  |  | 			} 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); | 
					
						
							|  |  |  | 				pp_dereverb_level = res_f; | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR,"Error! Preprocessor Dereverb Level must be >= 0\n"); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-04-08 21:40:57 +00:00
										 |  |  | 	ast_config_destroy(cfg); | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int reload(void)  | 
					
						
							| 
									
										
										
										
											2004-10-01 13:16:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	parse_config(); | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int unload_module(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	res = ast_unregister_translator(&lintospeex); | 
					
						
							| 
									
										
										
										
											2006-04-14 14:08:19 +00:00
										 |  |  | 	res |= ast_unregister_translator(&speextolin); | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int load_module(void) | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-01 03:18:48 +00:00
										 |  |  | 	parse_config(); | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	res=ast_register_translator(&speextolin); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	if (!res)  | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 		res=ast_register_translator(&lintospeex); | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		ast_unregister_translator(&speextolin); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2002-08-30 00:34:24 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2006-04-14 14:08:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "Speex Coder/Decoder", | 
					
						
							|  |  |  | 		.load = load_module, | 
					
						
							|  |  |  | 		.unload = unload_module, | 
					
						
							|  |  |  | 		.reload = reload, | 
					
						
							|  |  |  | 	       ); |