2
The mediastreamer library aims at providing modular media processing and I/O
3
for linphone, but also for any telephony application.
4
Copyright (C) 2001 Simon MORLAT simon.morlat@linphone.org
6
This library is free software; you can redistribute it and/or
7
modify it under the terms of the GNU Lesser General Public
8
License as published by the Free Software Foundation; either
9
version 2.1 of the License, or (at your option) any later version.
11
This library is distributed in the hope that it will be useful,
12
but WITHOUT ANY WARRANTY; without even the implied warranty of
13
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14
Lesser General Public License for more details.
16
You should have received a copy of the GNU Lesser General Public
17
License along with this library; if not, write to the Free Software
18
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23
#include "msspeexenc.h"
25
extern MSCodecInfo speex_info;
27
static MSSpeexEncClass * ms_speex_enc_class=NULL;
29
MSFilter * ms_speex_enc_new(SpeexMode *mode, gint quality)
31
MSSpeexEnc *obj=g_new(MSSpeexEnc,1);
33
if (ms_speex_enc_class==NULL){
34
ms_speex_enc_class=g_new(MSSpeexEncClass,1);
35
ms_speex_enc_class_init(ms_speex_enc_class,mode);
37
MS_FILTER(obj)->klass=MS_FILTER_CLASS(ms_speex_enc_class);
38
ms_speex_enc_init(MS_SPEEX_ENC(obj));
39
return MS_FILTER(obj);
42
void ms_speex_enc_init(MSSpeexEnc *obj)
44
ms_filter_init(MS_FILTER(obj));
45
MS_FILTER(obj)->infifos=obj->inf;
46
MS_FILTER(obj)->outqueues=obj->outq;
54
void ms_speex_enc_init_core(MSSpeexEnc *obj,SpeexMode *mode, gint bitrate)
56
int proc_type, proc_speed;
61
obj->speex_state=speex_encoder_init(mode);
62
speex_bits_init(&obj->bits);
66
speex_encoder_ctl(obj->speex_state, SPEEX_SET_BITRATE, &bitrate);
67
g_message("Setting speex output bitrate less or equal than %i",bitrate-1);
70
proc_speed=ms_proc_get_speed();
71
proc_vendor=ms_proc_get_param("vendor_id");
72
if (proc_speed<0 || proc_vendor==NULL){
73
g_warning("Can't guess processor features: setting speex encoder to its lowest complexity.");
75
speex_encoder_ctl(obj->speex_state,SPEEX_SET_COMPLEXITY,&tmp);
76
}else if ((proc_speed!=-1) && (proc_speed<200)){
77
g_warning("A cpu speed less than 200 Mhz is not enough: let's reduce the complexity of the speex codec.");
79
speex_encoder_ctl(obj->speex_state,SPEEX_SET_COMPLEXITY,&tmp);
80
}else if (proc_vendor!=NULL) {
81
if (strncmp(proc_vendor,"GenuineIntel",strlen("GenuineIntel"))==0){
82
proc_type=ms_proc_get_type();
84
g_warning("A pentium I is not enough fast for speex codec in normal mode: let's reduce its complexity.");
86
speex_encoder_ctl(obj->speex_state,SPEEX_SET_COMPLEXITY,&tmp);
91
/* guess the used input frame size */
92
speex_mode_query(mode, SPEEX_MODE_FRAME_SIZE, &frame_size);
93
MS_FILTER(obj)->r_mingran=frame_size*2;
94
ms_trace("ms_speex_init: using frame size of %i.",MS_FILTER(obj)->r_mingran);
96
obj->inbuf=g_malloc(sizeof(float)*frame_size);
100
/* must be called before the encoder is running*/
101
int ms_speex_enc_set_property(MSSpeexEnc *obj,int property,int *value)
103
if (obj->initialized){
104
/* we are called when speex is running !! forbid that! */
105
ms_warning("ms_speex_enc_set_property: cannot call this function when running!");
109
case MS_FILTER_PROPERTY_FREQ:
110
obj->frequency=value[0];
112
case MS_FILTER_PROPERTY_BITRATE: /* to specify max bitrate */
113
obj->bitrate=value[0];
119
void ms_speex_enc_setup(MSSpeexEnc *obj)
123
g_message("Speex encoder setup: freq=%i",obj->frequency);
124
if ( obj->frequency< 16000) mode=&speex_nb_mode;
125
else mode=&speex_wb_mode;
126
ms_speex_enc_init_core(obj,mode,obj->bitrate);
130
void ms_speex_enc_unsetup(MSSpeexEnc *obj)
132
ms_speex_enc_uninit_core(obj);
135
void ms_speex_enc_class_init(MSSpeexEncClass *klass, SpeexMode *mode)
139
ms_filter_class_init(MS_FILTER_CLASS(klass));
140
/* we take the larger (wb) frame size */
141
speex_mode_query(&speex_wb_mode, SPEEX_MODE_FRAME_SIZE, &frame_size);
142
MS_FILTER_CLASS(klass)->process=(MSFilterProcessFunc)ms_speex_enc_process;
143
MS_FILTER_CLASS(klass)->destroy=(MSFilterDestroyFunc)ms_speex_enc_destroy;
144
MS_FILTER_CLASS(klass)->setup=(MSFilterSetupFunc)ms_speex_enc_setup;
145
MS_FILTER_CLASS(klass)->unsetup=(MSFilterSetupFunc)ms_speex_enc_unsetup;
146
MS_FILTER_CLASS(klass)->set_property=(MSFilterPropertyFunc)ms_speex_enc_set_property;
147
ms_filter_class_set_name(MS_FILTER_CLASS(klass),"SpeexEncoder");
148
MS_FILTER_CLASS(klass)->info=(MSFilterInfo*)&speex_info;
149
MS_FILTER_CLASS(klass)->max_finputs=1;
150
MS_FILTER_CLASS(klass)->max_qoutputs=1;
151
MS_FILTER_CLASS(klass)->r_maxgran=frame_size*2;
152
ms_trace("ms_speex_enc_class_init: r_maxgran is %i.",MS_FILTER_CLASS(klass)->r_maxgran);
155
void ms_speex_enc_uninit_core(MSSpeexEnc *obj)
157
if (obj->initialized){
158
speex_encoder_destroy(obj->speex_state);
164
void ms_speex_enc_destroy(MSSpeexEnc *obj)
166
ms_speex_enc_uninit_core(obj);
170
void ms_speex_enc_process(MSSpeexEnc *obj)
172
MSFifo *inf=obj->inf[0];
173
MSQueue *outq=obj->outq[0];
175
gint gran=MS_FILTER(obj)->r_mingran;
179
g_return_if_fail(inf!=NULL);
180
g_return_if_fail(outq!=NULL);
182
ms_fifo_get_read_ptr(inf,gran,(void**)&input);
183
g_return_if_fail(input!=NULL);
184
/* convert into float */
185
for (i=0;i<gran/2;i++) obj->inbuf[i]=input[i];
187
speex_bits_reset(&obj->bits);
188
speex_encode(obj->speex_state,obj->inbuf,&obj->bits);
189
m=ms_message_new(speex_bits_nbytes(&obj->bits));
190
m->size=speex_bits_write(&obj->bits,m->data,m->size);
191
ms_queue_put(outq,m);