You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
248 lines
5.9 KiB
248 lines
5.9 KiB
/*
|
|
The mediastreamer library aims at providing modular media processing and I/O
|
|
for linphone, but also for any telephony application.
|
|
Copyright (C) 2001 Simon MORLAT simon.morlat@linphone.org
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "msosswrite.h"
|
|
#include "mssync.h"
|
|
#include <unistd.h>
|
|
#include <math.h>
|
|
|
|
MSFilterInfo oss_write_info={
|
|
"OSS write",
|
|
0,
|
|
MS_FILTER_OTHER,
|
|
ms_oss_write_new,
|
|
NULL
|
|
};
|
|
|
|
|
|
static MSOssWriteClass *msosswriteclass=NULL;
|
|
|
|
MSFilter * ms_oss_write_new()
|
|
{
|
|
MSOssWrite *w;
|
|
|
|
if (msosswriteclass==NULL)
|
|
{
|
|
msosswriteclass=g_new(MSOssWriteClass,1);
|
|
ms_oss_write_class_init( msosswriteclass );
|
|
}
|
|
w=g_new(MSOssWrite,1);
|
|
MS_FILTER(w)->klass=MS_FILTER_CLASS(msosswriteclass);
|
|
ms_oss_write_init(w);
|
|
return(MS_FILTER(w));
|
|
}
|
|
|
|
/* FOR INTERNAL USE*/
|
|
void ms_oss_write_init(MSOssWrite *w)
|
|
{
|
|
ms_sound_write_init(MS_SOUND_WRITE(w));
|
|
MS_FILTER(w)->infifos=w->f_inputs;
|
|
MS_FILTER(w)->infifos[0]=NULL;
|
|
MS_FILTER(w)->r_mingran=512; /* very few cards can do that...*/
|
|
w->devid=0;
|
|
w->sndcard=NULL;
|
|
w->freq=8000;
|
|
w->channels=1;
|
|
w->dtmf_time=-1;
|
|
}
|
|
|
|
gint ms_oss_write_set_property(MSOssWrite *f,MSFilterProperty prop, void *value)
|
|
{
|
|
switch(prop){
|
|
case MS_FILTER_PROPERTY_FREQ:
|
|
f->freq=((gint*)value)[0];
|
|
break;
|
|
case MS_FILTER_PROPERTY_CHANNELS:
|
|
f->channels=((gint*)value)[0];
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void ms_oss_write_class_init(MSOssWriteClass *klass)
|
|
{
|
|
ms_sound_write_class_init(MS_SOUND_WRITE_CLASS(klass));
|
|
MS_FILTER_CLASS(klass)->max_finputs=1; /* one fifo input only */
|
|
MS_FILTER_CLASS(klass)->r_maxgran=MS_OSS_WRITE_DEF_GRAN;
|
|
MS_FILTER_CLASS(klass)->process= (MSFilterProcessFunc)ms_oss_write_process;
|
|
MS_FILTER_CLASS(klass)->destroy= (MSFilterDestroyFunc)ms_oss_write_destroy;
|
|
MS_FILTER_CLASS(klass)->setup= (MSFilterSetupFunc)ms_oss_write_setup;
|
|
MS_FILTER_CLASS(klass)->unsetup= (MSFilterSetupFunc)ms_oss_write_stop;
|
|
MS_FILTER_CLASS(klass)->set_property=(MSFilterPropertyFunc)ms_oss_write_set_property;
|
|
MS_FILTER_CLASS(klass)->info=&oss_write_info;
|
|
MS_SOUND_WRITE_CLASS(klass)->set_device=(gint (*)(MSSoundWrite*,gint))ms_oss_write_set_device;
|
|
MS_SOUND_WRITE_CLASS(klass)->start=(void (*)(MSSoundWrite*))ms_oss_write_start;
|
|
MS_SOUND_WRITE_CLASS(klass)->stop=(void (*)(MSSoundWrite*))ms_oss_write_stop;
|
|
MS_SOUND_WRITE_CLASS(klass)->set_level=(void (*)(MSSoundWrite*, gint))ms_oss_write_set_level;
|
|
ms_filter_class_set_name(MS_FILTER_CLASS(klass),"OssWrite");
|
|
}
|
|
|
|
void ms_oss_write_destroy( MSOssWrite *obj)
|
|
{
|
|
|
|
g_free(obj);
|
|
}
|
|
|
|
void ms_oss_write_process(MSOssWrite *f)
|
|
{
|
|
MSFifo *fifo;
|
|
void *p;
|
|
int i;
|
|
gint gran=ms_filter_get_mingran(MS_FILTER(f));
|
|
|
|
/* always consume something */
|
|
fifo=f->f_inputs[0];
|
|
ms_fifo_get_read_ptr(fifo,gran,&p);
|
|
if (p==NULL) {
|
|
g_warning("Not enough data: gran=%i.",gran);
|
|
return;
|
|
}
|
|
g_return_if_fail(f->sndcard!=NULL);
|
|
if (f->dtmf_time!=-1){
|
|
gint16 *buf=(gint16*)p;
|
|
/* generate a DTMF*/
|
|
for(i=0;i<gran/2;i++){
|
|
buf[i]=(gint16)(10000.0*sin(2*M_PI*(double)f->dtmf_time*f->lowfreq));
|
|
buf[i]+=(gint16)(10000.0*sin(2*M_PI*(double)f->dtmf_time*f->highfreq));
|
|
f->dtmf_time++;
|
|
/* //printf("buf[%i]=%i\n",i,buf[i]); */
|
|
}
|
|
if (f->dtmf_time>f->dtmf_duration) f->dtmf_time=-1; /*finished*/
|
|
}
|
|
snd_card_write(f->sndcard,p,gran);
|
|
}
|
|
|
|
void ms_oss_write_start(MSOssWrite *w)
|
|
{
|
|
gint bsize;
|
|
g_return_if_fail(w->devid!=-1);
|
|
w->sndcard=snd_card_manager_get_card(snd_card_manager,w->devid);
|
|
g_return_if_fail(w->sndcard!=NULL);
|
|
/* open the device for an audio telephony signal with minimum latency */
|
|
snd_card_open_w(w->sndcard,16,w->channels==2,w->freq);
|
|
w->bsize=snd_card_get_bsize(w->sndcard);
|
|
/* //MS_FILTER(w)->r_mingran=w->bsize; */
|
|
/* //ms_sync_set_samples_per_tick(MS_FILTER(w)->sync,bsize); */
|
|
}
|
|
|
|
void ms_oss_write_stop(MSOssWrite *w)
|
|
{
|
|
g_return_if_fail(w->devid!=-1);
|
|
g_return_if_fail(w->sndcard!=NULL);
|
|
snd_card_close_w(w->sndcard);
|
|
w->sndcard=NULL;
|
|
}
|
|
|
|
void ms_oss_write_set_level(MSOssWrite *w,gint a)
|
|
{
|
|
|
|
}
|
|
|
|
gint ms_oss_write_set_device(MSOssWrite *w, gint devid)
|
|
{
|
|
w->devid=devid;
|
|
return 0;
|
|
}
|
|
|
|
void ms_oss_write_setup(MSOssWrite *r)
|
|
{
|
|
/* //g_message("starting MSOssWrite.."); */
|
|
ms_oss_write_start(r);
|
|
}
|
|
|
|
|
|
|
|
void ms_oss_write_play_dtmf(MSOssWrite *w, char dtmf){
|
|
|
|
w->dtmf_duration=0.1*w->freq;
|
|
switch(dtmf){
|
|
case '0':
|
|
w->lowfreq=941;
|
|
w->highfreq=1336;
|
|
break;
|
|
case '1':
|
|
w->lowfreq=697;
|
|
w->highfreq=1209;
|
|
break;
|
|
case '2':
|
|
w->lowfreq=697;
|
|
w->highfreq=1336;
|
|
break;
|
|
case '3':
|
|
w->lowfreq=697;
|
|
w->highfreq=1477;
|
|
break;
|
|
case '4':
|
|
w->lowfreq=770;
|
|
w->highfreq=1209;
|
|
break;
|
|
case '5':
|
|
w->lowfreq=770;
|
|
w->highfreq=1336;
|
|
break;
|
|
case '6':
|
|
w->lowfreq=770;
|
|
w->highfreq=1477;
|
|
break;
|
|
case '7':
|
|
w->lowfreq=852;
|
|
w->highfreq=1209;
|
|
break;
|
|
case '8':
|
|
w->lowfreq=852;
|
|
w->highfreq=1336;
|
|
break;
|
|
case '9':
|
|
w->lowfreq=852;
|
|
w->highfreq=1477;
|
|
break;
|
|
case '*':
|
|
w->lowfreq=941;
|
|
w->highfreq=1209;
|
|
break;
|
|
case '#':
|
|
w->lowfreq=941;
|
|
w->highfreq=1477;
|
|
break;
|
|
case 'A':
|
|
w->lowfreq=697;
|
|
w->highfreq=1633;
|
|
break;
|
|
case 'B':
|
|
w->lowfreq=770;
|
|
w->highfreq=1633;
|
|
break;
|
|
case 'C':
|
|
w->lowfreq=852;
|
|
w->highfreq=1633;
|
|
break;
|
|
case 'D':
|
|
w->lowfreq=941;
|
|
w->highfreq=1633;
|
|
break;
|
|
default:
|
|
g_warning("Not a dtmf key.");
|
|
return;
|
|
}
|
|
w->lowfreq=w->lowfreq/w->freq;
|
|
w->highfreq=w->highfreq/w->freq;
|
|
w->dtmf_time=0;
|
|
}
|