3 Copyright (C) 2010 Simon MORLAT (simon.morlat@linphone.org)
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.
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.
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.
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.
25 #include "linphonecore_utils.h"
27 #include "mediastreamer2/msticker.h"
28 #include "mediastreamer2/mssndcard.h"
29 #include "mediastreamer2/msaudiomixer.h"
30 #include "mediastreamer2/mschanadapter.h"
31 #include "mediastreamer2/msfileplayer.h"
32 #include "mediastreamer2/msitc.h"
35 static struct _MSSndCard *linphone_sound_daemon_get_proxy_card(LinphoneSoundDaemon *obj);
37 #define MAX_BRANCHES 10
41 struct _LinphoneSoundDaemon *lsd;
44 MSFilter *chanadapter;
45 LsdEndOfPlayCallback eop_cb;
52 struct _LinphoneSoundDaemon {
60 LsdPlayer branches[MAX_BRANCHES];
63 static MSFilter *create_writer(MSSndCard *c){
64 LinphoneSoundDaemon *lsd=(LinphoneSoundDaemon*)c->data;
65 lsd->itcsink=ms_filter_new(MS_ITC_SINK_ID);
66 ms_filter_call_method(lsd->itcsink,MS_ITC_SINK_CONNECT,lsd->branches[0].player);
70 static MSSndCardDesc proxycard={
71 "Linphone Sound Daemon",
79 /*create_reader*/ NULL,
84 LsdPlayer *linphone_sound_daemon_get_player(LinphoneSoundDaemon *obj){
86 for(i=1;i<MAX_BRANCHES;++i){
87 LsdPlayer *b=&obj->branches[i];
88 MSFilter *p=b->player;
90 ms_filter_call_method(p,MS_PLAYER_GET_STATE,&state);
91 if (state==MSPlayerClosed){
92 lsd_player_set_gain(b,1);
93 lsd_player_enable_loop (b,FALSE);
97 ms_warning("No more free players !");
101 void linphone_sound_daemon_release_player(LinphoneSoundDaemon *obj, LsdPlayer * player){
103 ms_filter_call_method(player->player,MS_PLAYER_GET_STATE,&state);
104 if (state!=MSPlayerClosed){
105 ms_filter_call_method(player->player,MS_PLAYER_CLOSE,&state);
109 LinphoneSoundDaemon *lsd_player_get_daemon(const LsdPlayer *p){
113 int lsd_player_stop(LsdPlayer *p){
114 ms_filter_call_method_noarg(p->player,MS_PLAYER_PAUSE);
118 static void lsd_player_init(LsdPlayer *p, MSConnectionPoint mixer, MSFilterId playerid, LinphoneSoundDaemon *lsd){
119 MSConnectionHelper h;
120 p->player=ms_filter_new(playerid);
121 p->rateconv=ms_filter_new(MS_RESAMPLE_ID);
122 p->chanadapter=ms_filter_new(MS_CHANNEL_ADAPTER_ID);
124 ms_connection_helper_start(&h);
125 ms_connection_helper_link(&h,p->player,-1,0);
126 ms_connection_helper_link(&h,p->rateconv,0,0);
127 ms_connection_helper_link(&h,p->chanadapter,0,0);
128 ms_connection_helper_link(&h,mixer.filter,mixer.pin,-1);
129 p->mixer_pin=mixer.pin;
134 static void lsd_player_uninit(LsdPlayer *p, MSConnectionPoint mixer){
135 MSConnectionHelper h;
137 ms_connection_helper_start(&h);
138 ms_connection_helper_unlink (&h,p->player,-1,0);
139 ms_connection_helper_unlink(&h,p->rateconv,0,0);
140 ms_connection_helper_unlink(&h,p->chanadapter,0,0);
141 ms_connection_helper_unlink(&h,mixer.filter,mixer.pin,-1);
143 ms_filter_destroy(p->player);
144 ms_filter_destroy(p->rateconv);
145 ms_filter_destroy(p->chanadapter);
148 void lsd_player_set_callback(LsdPlayer *p, LsdEndOfPlayCallback cb){
152 void lsd_player_set_user_pointer(LsdPlayer *p, void *up){
156 void *lsd_player_get_user_pointer(const LsdPlayer *p){
160 static void lsd_player_on_eop(void * userdata, unsigned int id, void *arg){
161 LsdPlayer *p=(LsdPlayer *)userdata;
166 int lsd_player_play(LsdPlayer *b, const char *filename ){
169 LinphoneSoundDaemon *lsd=b->lsd;
171 ms_filter_call_method(b->player,MS_PLAYER_GET_STATE,&state);
172 if (state!=MSPlayerClosed){
173 ms_filter_call_method_noarg(b->player,MS_PLAYER_CLOSE);
176 if (ms_filter_call_method(b->player,MS_PLAYER_OPEN,(void*)filename)!=0){
177 ms_warning("Could not play %s",filename);
180 ms_filter_call_method(b->player,MS_FILTER_GET_SAMPLE_RATE,&rate);
181 ms_filter_call_method(b->player,MS_FILTER_GET_NCHANNELS,&chans);
182 ms_filter_set_notify_callback (b->player,lsd_player_on_eop,b);
184 ms_filter_call_method(b->rateconv,MS_FILTER_SET_SAMPLE_RATE,&rate);
185 ms_filter_call_method(b->rateconv,MS_FILTER_SET_NCHANNELS,&chans);
186 ms_filter_call_method(b->rateconv,MS_FILTER_SET_OUTPUT_SAMPLE_RATE,&lsd->out_rate);
188 ms_filter_call_method(b->chanadapter,MS_FILTER_SET_NCHANNELS,&chans);
189 ms_filter_call_method(b->chanadapter,MS_CHANNEL_ADAPTER_SET_OUTPUT_NCHANNELS,&lsd->out_nchans);
190 ms_filter_call_method_noarg (b->player,MS_PLAYER_START);
194 void lsd_player_enable_loop(LsdPlayer *p, bool_t loopmode){
195 if (ms_filter_get_id(p->player)==MS_FILE_PLAYER_ID){
196 int arg=loopmode ? 0 : -1;
197 ms_filter_call_method(p->player,MS_FILE_PLAYER_LOOP,&arg);
202 bool_t lsd_player_loop_enabled(const LsdPlayer *p){
206 void lsd_player_set_gain(LsdPlayer *p, float gain){
207 MSAudioMixerCtl gainctl;
208 gainctl.pin=p->mixer_pin;
210 ms_filter_call_method(p->lsd->mixer,MS_AUDIO_MIXER_SET_INPUT_GAIN,&gainctl);
213 LinphoneSoundDaemon * linphone_sound_daemon_new(const char *cardname){
215 MSConnectionPoint mp;
216 LinphoneSoundDaemon *lsd;
217 MSSndCard *card=ms_snd_card_manager_get_card(
218 ms_snd_card_manager_get(),
221 card=ms_snd_card_manager_get_default_playback_card (
222 ms_snd_card_manager_get());
224 ms_error("linphone_sound_daemon_new(): No playback soundcard available");
229 lsd=ms_new0(LinphoneSoundDaemon,1);
230 lsd->soundout=ms_snd_card_create_writer(card);
231 lsd->mixer=ms_filter_new(MS_AUDIO_MIXER_ID);
232 lsd->itcsink=ms_filter_new(MS_ITC_SINK_ID);
235 ms_filter_call_method(lsd->soundout,MS_FILTER_SET_SAMPLE_RATE,&lsd->out_rate);
236 ms_filter_call_method(lsd->soundout,MS_FILTER_SET_NCHANNELS,&lsd->out_nchans);
237 ms_filter_call_method(lsd->mixer,MS_FILTER_SET_SAMPLE_RATE,&lsd->out_rate);
238 ms_filter_call_method(lsd->mixer,MS_FILTER_SET_NCHANNELS,&lsd->out_nchans);
240 mp.filter=lsd->mixer;
243 lsd_player_init(&lsd->branches[0],mp,MS_ITC_SOURCE_ID,lsd);
244 for(i=1;i<MAX_BRANCHES;++i){
246 lsd_player_init(&lsd->branches[i],mp,MS_FILE_PLAYER_ID,lsd);
248 ms_filter_link(lsd->mixer,0,lsd->soundout,0);
249 lsd->ticker=ms_ticker_new();
250 ms_ticker_attach(lsd->ticker,lsd->soundout);
252 lsd->proxycard=ms_snd_card_new(&proxycard);
253 lsd->proxycard->data=lsd;
258 void linphone_sound_daemon_destroy(LinphoneSoundDaemon *obj){
260 MSConnectionPoint mp;
261 ms_ticker_detach(obj->ticker,obj->soundout);
262 mp.filter=obj->mixer;
263 for(i=0;i<MAX_BRANCHES;++i){
265 lsd_player_uninit (&obj->branches[i],mp);
267 ms_filter_unlink(obj->mixer,0,obj->soundout,0);
268 ms_ticker_destroy(obj->ticker);
269 ms_filter_destroy(obj->soundout);
270 ms_filter_destroy(obj->mixer);
271 ms_filter_destroy(obj->itcsink);
274 MSSndCard *linphone_sound_daemon_get_proxy_card(LinphoneSoundDaemon *lsd){
275 return lsd->proxycard;
278 void linphone_core_use_sound_daemon(LinphoneCore *lc, LinphoneSoundDaemon *lsd){
279 lc->sound_conf.lsd_card=linphone_sound_daemon_get_proxy_card (lsd);