]> sjero.net Git - linphone/blob - coreapi/lsd.c
cb0f0af46033924321725d9b46707ea5e2412ac4
[linphone] / coreapi / lsd.c
1 /*
2 linphone
3 Copyright (C) 2010 Simon MORLAT (simon.morlat@linphone.org)
4
5 This program is free software; you can redistribute it and/or
6 modify it under the terms of the GNU General Public License
7 as published by the Free Software Foundation; either version 2
8 of the License, or (at your option) any later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
18 */
19
20 /* Linphone Sound Daemon: is a lightweight utility to play sounds to speaker during a conversation.
21  This is useful for embedded platforms, where sound apis are not performant enough to allow
22  simultaneous sound access.
23 */
24
25 #include "linphonecore_utils.h"
26 #include "mediastreamer2/msticker.h"
27 #include "mediastreamer2/mssndcard.h"
28 #include "mediastreamer2/msaudiomixer.h"
29 #include "mediastreamer2/mschanadapter.h"
30 #include "mediastreamer2/msfileplayer.h"
31 #include "mediastreamer2/msitc.h"
32
33
34
35 #define MAX_BRANCHES 10
36
37
38 struct _LsdPlayer{
39         struct _LinphoneSoundDaemon *lsd;
40         MSFilter *player;
41         MSFilter *rateconv;
42         MSFilter *chanadapter;
43         LsdEndOfPlayCallback eop_cb;
44         int mixer_pin;
45         void *user_data;
46 };
47
48 struct _LinphoneSoundDaemon {
49         int out_rate;
50         int out_nchans;
51         MSFilter *mixer;
52         MSFilter *soundout;
53         MSFilter *itcsink;
54         MSTicker *ticker;
55         MSSndCard *proxycard;
56         LsdPlayer branches[MAX_BRANCHES];
57 };
58
59 static MSFilter *create_writer(MSSndCard *c){
60         LinphoneSoundDaemon *lsd=(LinphoneSoundDaemon*)c->data;
61         return lsd->itcsink;
62 }
63
64 static MSSndCardDesc proxycard={
65         "Linphone Sound Daemon",
66         /*detect*/ NULL,
67         /*init*/ NULL,
68         NULL,
69         NULL,
70         NULL,
71         NULL,
72         NULL,
73         /*create_reader*/ NULL,
74         create_writer,
75         /*uninit,*/
76 };
77
78 LsdPlayer *linphone_sound_daemon_get_player(LinphoneSoundDaemon *obj){
79         int i;
80         for(i=0;i<MAX_BRANCHES;++i){
81                 LsdPlayer *b=&obj->branches[i];
82                 MSFilter *p=b->player;
83                 int state;
84                 ms_filter_call_method(p,MS_PLAYER_GET_STATE,&state);
85                 if (state==MSPlayerClosed){
86                         lsd_player_set_gain(b,1);
87                         return b;
88                 }
89         }
90         ms_warning("No more free players !");
91         return NULL;
92 }
93
94 void linphone_sound_daemon_release_player(LinphoneSoundDaemon *obj, LsdPlayer * player){
95         int state;
96         ms_filter_call_method(player->player,MS_PLAYER_GET_STATE,&state);
97         if (state!=MSPlayerClosed){
98                 ms_filter_call_method(player->player,MS_PLAYER_CLOSE,&state);
99         }
100 }
101
102 int lsd_player_stop(LsdPlayer *p){
103         ms_filter_call_method_noarg(p->player,MS_PLAYER_PAUSE);
104         return 0;
105 }
106
107 static void lsd_player_init(LsdPlayer *p, MSConnectionPoint mixer, MSFilterId playerid, LinphoneSoundDaemon *lsd){
108         MSConnectionHelper h;
109         p->player=ms_filter_new(playerid);
110         p->rateconv=ms_filter_new(MS_RESAMPLE_ID);
111         p->chanadapter=ms_filter_new(MS_CHANNEL_ADAPTER_ID);
112         
113         ms_connection_helper_start(&h);
114         ms_connection_helper_link(&h,p->player,-1,0);
115         ms_connection_helper_link(&h,p->rateconv,0,0);
116         ms_connection_helper_link(&h,p->chanadapter,0,0);
117         ms_connection_helper_link(&h,mixer.filter,mixer.pin,-1);
118         p->mixer_pin=mixer.pin;
119         p->lsd=lsd;
120 }
121
122 static void lsd_player_uninit(LsdPlayer *p, MSConnectionPoint mixer){
123         MSConnectionHelper h;
124
125         ms_connection_helper_start(&h);
126         ms_connection_helper_unlink (&h,p->player,-1,0);
127         ms_connection_helper_unlink(&h,p->rateconv,0,0);
128         ms_connection_helper_unlink(&h,p->chanadapter,0,0);
129         ms_connection_helper_unlink(&h,mixer.filter,mixer.pin,-1);
130
131         ms_filter_destroy(p->player);
132         ms_filter_destroy(p->rateconv);
133         ms_filter_destroy(p->chanadapter);
134 }
135
136 void lsd_player_set_callback(LsdPlayer *p, LsdEndOfPlayCallback cb){
137         p->eop_cb=cb;
138 }
139
140 void lsd_player_set_user_pointer(LsdPlayer *p, void *up){
141         p->user_data=up;
142 }
143
144 void *lsd_player_get_user_pointer(LsdPlayer *p){
145         return p->user_data;
146 }
147
148 static void lsd_player_on_eop(void * userdata, unsigned int id, void *arg){
149 }
150
151 int lsd_player_play(LsdPlayer *b, const char *filename ){
152         int rate,chans;
153         int state;
154         LinphoneSoundDaemon *lsd=b->lsd;
155         
156         ms_filter_call_method(b->player,MS_PLAYER_GET_STATE,&state);
157         if (state!=MSPlayerClosed){
158                 ms_filter_call_method_noarg(b->player,MS_PLAYER_CLOSE);
159         }
160         
161         if (ms_filter_call_method(b->player,MS_PLAYER_OPEN,(void*)filename)!=0){
162                 return -1;
163         }
164         ms_filter_call_method(b->player,MS_FILTER_GET_SAMPLE_RATE,&rate);
165         ms_filter_call_method(b->player,MS_FILTER_GET_NCHANNELS,&chans);
166         ms_filter_set_notify_callback (b->player,lsd_player_on_eop,b);
167         
168         ms_filter_call_method(b->rateconv,MS_FILTER_SET_SAMPLE_RATE,&rate);
169         ms_filter_call_method(b->rateconv,MS_FILTER_SET_NCHANNELS,&chans);
170         ms_filter_call_method(b->rateconv,MS_FILTER_SET_OUTPUT_SAMPLE_RATE,&lsd->out_rate);
171
172         ms_filter_call_method(b->chanadapter,MS_FILTER_SET_NCHANNELS,&chans);
173         ms_filter_call_method(b->chanadapter,MS_CHANNEL_ADAPTER_SET_OUTPUT_NCHANNELS,&lsd->out_nchans);
174         return 0;
175 }
176
177 int lsd_player_stop(LsdPlayer *p);
178
179 void lsd_player_enable_loop(LsdPlayer *p, bool_t loopmode){
180         if (ms_filter_get_id(p->player)==MS_FILE_PLAYER_ID){
181                 int arg=loopmode ? 0 : -1;
182                 ms_filter_call_method(p->player,MS_FILE_PLAYER_LOOP,&arg);
183         }
184 }
185
186 void lsd_player_set_gain(LsdPlayer *p, float gain){
187         MSAudioMixerCtl gainctl;
188         gainctl.pin=p->mixer_pin;
189         gainctl.gain=gain;
190         ms_filter_call_method(p->lsd->mixer,MS_AUDIO_MIXER_SET_INPUT_GAIN,&gainctl);
191 }
192
193 LinphoneSoundDaemon * linphone_sound_daemon_new(const char *cardname){
194         int i;
195         MSConnectionPoint mp;
196         LinphoneSoundDaemon *lsd;
197         MSSndCard *card=ms_snd_card_manager_get_card(
198                                                      ms_snd_card_manager_get(),
199                                                      cardname);
200         if (card==NULL){
201                 card=ms_snd_card_manager_get_default_playback_card (
202                                                                     ms_snd_card_manager_get());
203                 if (card==NULL){
204                         ms_error("linphone_sound_daemon_new(): No playback soundcard available");
205                         return NULL;
206                 }
207         }
208         
209         lsd=ms_new0(LinphoneSoundDaemon,1);
210         lsd->soundout=ms_snd_card_create_writer(card);
211         lsd->mixer=ms_filter_new(MS_AUDIO_MIXER_ID);
212         lsd->itcsink=ms_filter_new(MS_ITC_SINK_ID);
213         lsd->out_rate=44100;
214         lsd->out_nchans=2;
215         ms_filter_call_method(lsd->soundout,MS_FILTER_SET_SAMPLE_RATE,&lsd->out_rate);
216         ms_filter_call_method(lsd->soundout,MS_FILTER_SET_NCHANNELS,&lsd->out_nchans);
217         ms_filter_call_method(lsd->mixer,MS_FILTER_SET_SAMPLE_RATE,&lsd->out_rate);
218         ms_filter_call_method(lsd->mixer,MS_FILTER_SET_NCHANNELS,&lsd->out_nchans);
219
220         mp.filter=lsd->mixer;
221         mp.pin=0;
222
223         lsd_player_init(&lsd->branches[0],mp,MS_ITC_SINK_ID,lsd);
224         for(i=1;i<MAX_BRANCHES;++i){
225                 mp.pin=i;
226                 lsd_player_init(&lsd->branches[i],mp,MS_FILE_PLAYER_ID,lsd);
227         }
228         ms_filter_link(lsd->mixer,0,lsd->soundout,0);
229         lsd->ticker=ms_ticker_new();
230         ms_ticker_attach(lsd->ticker,lsd->soundout);
231
232         lsd->proxycard=ms_snd_card_new(&proxycard);
233         lsd->proxycard->data=lsd;
234
235         ms_filter_call_method(lsd->itcsink,MS_ITC_SINK_CONNECT,lsd->branches[0].player);
236         
237         return lsd;
238 }
239
240 void linphone_sound_daemon_destroy(LinphoneSoundDaemon *obj){
241         int i;
242         MSConnectionPoint mp;
243         ms_ticker_detach(obj->ticker,obj->soundout);
244         mp.filter=obj->mixer;
245         for(i=0;i<MAX_BRANCHES;++i){
246                 mp.pin=i;
247                 lsd_player_uninit (&obj->branches[i],mp);
248         }
249         ms_ticker_destroy(obj->ticker);
250         ms_filter_destroy(obj->soundout);
251         ms_filter_destroy(obj->mixer);
252         ms_filter_destroy(obj->itcsink);
253 }
254
255
256
257 MSSndCard *linphone_sound_daemon_get_proxy_card(LinphoneSoundDaemon *lsd){
258         return lsd->proxycard;
259 }