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
21
#include "msringplayer.h"
25
#include <sys/types.h>
29
#include "waveheader.h"
31
#define WAVE_HEADER_OFFSET sizeof(wave_header_t)
33
enum { PLAY_RING, PLAY_SILENCE};
35
static int supported_freq[6]={8000,11025,16000,22050,32000,44100};
37
gint freq_is_supported(gint freq){
40
if (abs(supported_freq[i]-freq)<50) return supported_freq[i];
45
static MSRingPlayerClass *ms_ring_player_class=NULL;
49
* @name: The path to the 16-bit 8khz raw file to be played as a ring.
50
* @seconds: The number of seconds that separates two rings.
52
* Allocates a new MSRingPlayer object.
55
* Returns: a pointer the the object, NULL if name could not be open.
57
MSFilter * ms_ring_player_new(char *name, gint seconds)
62
if ((name!=NULL) && (strlen(name)!=0))
64
fd=open(name,O_RDONLY);
67
g_warning("ms_ring_player_new: failed to open %s.\n",name);
72
g_warning("ms_ring_player_new: Bad file name");
76
r=g_new(MSRingPlayer,1);
77
ms_ring_player_init(r);
78
if (ms_ring_player_class==NULL)
80
ms_ring_player_class=g_new(MSRingPlayerClass,1);
81
ms_ring_player_class_init(ms_ring_player_class);
83
MS_FILTER(r)->klass=MS_FILTER_CLASS(ms_ring_player_class);
88
if (strstr(name,".wav")!=NULL){
92
read(fd,&header,sizeof(wave_header_t));
93
freq=wave_header_get_rate(&header);
94
if ((freq2=freq_is_supported(freq))>0){
97
g_warning("Unsupported sampling rate %i",freq);
100
r->channel=wave_header_get_channel(&header);
101
lseek(fd,WAVE_HEADER_OFFSET,SEEK_SET);
102
#ifdef WORDS_BIGENDIAN
108
ms_ring_player_set_property(r, MS_FILTER_PROPERTY_FREQ,&r->freq);
110
return(MS_FILTER(r));
114
/* FOR INTERNAL USE*/
115
void ms_ring_player_init(MSRingPlayer *r)
117
ms_filter_init(MS_FILTER(r));
118
MS_FILTER(r)->outfifos=r->foutputs;
119
MS_FILTER(r)->outqueues=r->qoutputs;
120
memset(r->foutputs,0,sizeof(MSFifo*)*MS_RING_PLAYER_MAX_OUTPUTS);
121
memset(r->qoutputs,0,sizeof(MSQueue*)*MS_RING_PLAYER_MAX_OUTPUTS);
128
gint ms_ring_player_set_property(MSRingPlayer *f,MSFilterProperty prop, void *value)
131
case MS_FILTER_PROPERTY_FREQ:
132
f->rate=((gint*)value)[0]*2;
133
f->silence_bytes=f->silence*f->rate;
135
f->gran=(f->rate*f->sync->interval/1000)*2;
141
gint ms_ring_player_get_property(MSRingPlayer *f,MSFilterProperty prop, void *value)
144
case MS_FILTER_PROPERTY_FREQ:
145
((gint*)value)[0]=f->freq;
148
case MS_FILTER_PROPERTY_CHANNELS:
149
((gint*)value)[0]=f->channel;
155
gint ms_ring_player_get_sample_freq(MSRingPlayer *obj){
160
void ms_ring_player_class_init(MSRingPlayerClass *klass)
162
ms_filter_class_init(MS_FILTER_CLASS(klass));
163
ms_filter_class_set_name(MS_FILTER_CLASS(klass),"ringplay");
164
ms_filter_class_set_attr(MS_FILTER_CLASS(klass),FILTER_IS_SOURCE);
165
MS_FILTER_CLASS(klass)->max_foutputs=MS_RING_PLAYER_MAX_OUTPUTS;
166
MS_FILTER_CLASS(klass)->max_qoutputs=MS_RING_PLAYER_MAX_OUTPUTS;
167
MS_FILTER_CLASS(klass)->w_maxgran=MS_RING_PLAYER_DEF_GRAN;
168
MS_FILTER_CLASS(klass)->setup=(MSFilterSetupFunc)ms_ring_player_setup;
169
MS_FILTER_CLASS(klass)->destroy=(MSFilterDestroyFunc)ms_ring_player_destroy;
170
MS_FILTER_CLASS(klass)->process=(MSFilterProcessFunc)ms_ring_player_process;
171
MS_FILTER_CLASS(klass)->set_property=(MSFilterPropertyFunc)ms_ring_player_set_property;
172
MS_FILTER_CLASS(klass)->get_property=(MSFilterPropertyFunc)ms_ring_player_get_property;
175
void ms_ring_player_process(MSRingPlayer *r)
183
g_return_if_fail(gran>0);
184
/* process output fifos*/
187
ms_fifo_get_write_ptr(f,gran,(void**)&p);
188
g_return_if_fail(p!=NULL);
189
for (processed=0;processed<gran;){
192
err=read(r->fd,&p[processed],gran-processed);
195
memset(&p[processed],0,gran-processed);
197
g_warning("ms_ring_player_process: failed to read: %s.\n",strerror(errno));
203
r->current_pos=r->silence_bytes;
204
lseek(r->fd,WAVE_HEADER_OFFSET,SEEK_SET);
205
r->state=PLAY_SILENCE;
206
ms_filter_notify_event(MS_FILTER(r),MS_RING_PLAYER_END_OF_RING_EVENT,NULL);
208
if (r->need_swap) swap_buffer(&p[processed],err);
213
if (r->current_pos>err){
214
memset(&p[processed],0,err);
215
r->current_pos-=gran;
218
memset(&p[processed],0,r->current_pos);
219
processed+=r->current_pos;
228
* ms_ring_player_destroy:
229
* @obj: A valid MSRingPlayer object.
231
* Destroy a MSRingPlayer object.
236
void ms_ring_player_destroy( MSRingPlayer *obj)
238
if (obj->fd!=0) close(obj->fd);
242
void ms_ring_player_setup(MSRingPlayer *r,MSSync *sync)
245
r->gran=(r->rate*r->sync->interval/1000)*r->channel;