2 * Remix Stream Player: TIZEN device output
4 * Govindaraju SM <govi.sm@samsung.com>, August 2011
5 * Prince Kumar Dubey <prince.dubey@samsung.com>, August 2011
10 #include <remix/remix.h>
13 #ifdef HAVE_LIBSNDFILE
18 int _edje_multisense_default_log_dom = -1;
23 #define ERR(...) EINA_LOG_DOM_ERR(_edje_multisense_default_log_dom, __VA_ARGS__)
27 #define WRN(...) EINA_LOG_DOM_WARN(_edje_multisense_default_log_dom, __VA_ARGS__)
29 #define STREAM_PLAYER_BUFFERLEN 2048
30 #define DEFAULT_FORMAT MMSOUND_PCM_S16_LE
31 #define TIMEOUT_FOR_MM_HANDLER 10
33 typedef struct _RemixPlayerData RemixPlayerData;
34 typedef short PLAYER_PCM;
36 struct _RemixPlayerData {
37 RemixPCM databuffer[STREAM_PLAYER_BUFFERLEN];
38 PLAYER_PCM *playbuffer;
39 MMSoundPcmHandle_t handle;
40 MMSoundPcmChannel_t channel;
53 /* Optimisation dependencies: none */
54 static RemixBase *remix_player_optimise (RemixEnv *env, RemixBase *base);
56 //Tizen Only : Remove API definition for avoiding build warning
59 remix_player_reset_device (RemixEnv *env, RemixBase *base)
61 RemixPlayerData *player_data = remix_base_get_instance_data(env, base);
63 if (player_data->handle)
65 if (mm_sound_pcm_play_stop(player_data->handle) < 0)
66 remix_set_error (env, REMIX_ERROR_SYSTEM);
68 mm_sound_pcm_play_close(player_data->handle);
69 player_data->handle = NULL;
71 player_data->old_buffer_size = player_data->buffer_size;
72 player_data->buffer_size = mm_sound_pcm_play_open_no_session(&player_data->handle,
73 player_data->frequency,
77 if (mm_sound_pcm_play_start(player_data->handle) < 0)
79 remix_set_error (env, REMIX_ERROR_SYSTEM);
83 if (player_data->old_buffer_size < player_data->buffer_size)
85 if ( player_data->playbuffer ) free(player_data->playbuffer);
86 player_data->playbuffer = calloc(sizeof(PLAYER_PCM), player_data->buffer_size);
87 if (!player_data->playbuffer)
89 remix_set_error (env, REMIX_ERROR_SYSTEM);
98 _vconf_noti_key_changed_cb(keynode_t *node, void *data)
100 RemixPlayerData *player = (RemixPlayerData *)data;
101 char *keyname = vconf_keynode_get_name(node);
103 if (strcmp(keyname, VCONFKEY_SETAPPL_SOUND_STATUS_BOOL) == 0)
104 player->snd_on = vconf_keynode_get_bool(node);
106 if (strcmp(keyname, VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL) == 0)
107 player->tch_snd_on = vconf_keynode_get_bool(node);
111 remix_player_init (RemixEnv *env, RemixBase *base, CDSet *parameters)
113 RemixCount nr_channels;
115 RemixPlayerData *player_data = calloc(1, sizeof (RemixPlayerData));
119 remix_set_error(env, REMIX_ERROR_SYSTEM);
123 remix_base_set_instance_data(env, base, player_data);
124 channels = remix_get_channels (env);
126 nr_channels = cd_set_size (env, channels);
127 if (nr_channels == 1)
129 player_data->stereo = 0;
130 player_data->channel = MMSOUND_PCM_MONO;
132 else if (nr_channels == 2)
134 player_data->stereo = 1;
135 player_data->channel = MMSOUND_PCM_STEREO;
138 player_data->frequency = remix_get_samplerate(env);
139 player_data->buffer_size = 0;
140 player_data->max_value = (RemixPCM) SHRT_MAX / 2;
142 base = remix_player_optimise (env, base);
143 player_data->env = env;
144 player_data->base = base;
146 if (vconf_get_bool(VCONFKEY_SETAPPL_SOUND_STATUS_BOOL,
147 &player_data->snd_on) < 0)
148 WRN("\nFail to get VCONFKEY_SETAPPL_SOUND_STATUS_BOOL boolean value");
150 if (vconf_get_bool(VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL,
151 &player_data->tch_snd_on) < 0)
152 WRN("\nFail to get VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL boolean value");
154 if (vconf_notify_key_changed(VCONFKEY_SETAPPL_SOUND_STATUS_BOOL,
155 _vconf_noti_key_changed_cb, player_data) < 0)
156 WRN("\nFail to register VCONFKEY_SETAPPL_SOUND_STATUS_BOOL key callback");
158 if (vconf_notify_key_changed(VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL,
159 _vconf_noti_key_changed_cb, player_data) < 0)
160 WRN("\nFail to register VCONFKEY_SETAPPL_SOUND_STATUS_BOOL key callback");
162 if (vconf_notify_key_changed(VCONFKEY_SYSMAN_EARJACK,
163 _vconf_noti_key_changed_cb, player_data) < 0)
164 WRN("\nFail to register VCONFKEY_SYSMAN_EARJACK key callback");
171 remix_player_clone (RemixEnv *env, RemixBase *base)
173 RemixBase *new_player = remix_base_new (env);
175 remix_player_init( env, new_player, NULL);
180 remix_player_destroy (RemixEnv *env, RemixBase *base)
182 RemixPlayerData *player_data = remix_base_get_instance_data(env, base);
184 if (vconf_ignore_key_changed(VCONFKEY_SETAPPL_SOUND_STATUS_BOOL,
185 _vconf_noti_key_changed_cb) < 0)
186 WRN("\nFail to unregister VCONFKEY_SETAPPL_SOUND_STATUS_BOOL key callback");
187 if (vconf_ignore_key_changed(VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL,
188 _vconf_noti_key_changed_cb) < 0)
189 WRN("\nFail to unregister VCONFKEY_SETAPPL_TOUCH_SOUNDS_BOOL key callback");
190 if (vconf_ignore_key_changed(VCONFKEY_SYSMAN_EARJACK,
191 _vconf_noti_key_changed_cb) < 0)
192 WRN("\nFail to unregister VCONFKEY_SYSMAN_EARJACK key callback");
194 if (player_data->handle)
196 if (mm_sound_pcm_play_stop(player_data->handle) < 0)
197 remix_set_error (env, REMIX_ERROR_SYSTEM);
199 mm_sound_pcm_play_close(player_data->handle);
200 player_data->handle = NULL;
202 if (player_data->timeout) ecore_timer_del(player_data->timeout);
203 if (player_data->playbuffer) free(player_data->playbuffer);
209 remix_player_ready (RemixEnv *env, RemixBase *base)
211 RemixPlayerData *player_data = remix_base_get_instance_data(env, base);
212 RemixCount nr_channels;
216 channels = remix_get_channels (env);
217 samplerate = (int) remix_get_samplerate (env);
219 nr_channels = cd_set_size (env, channels);
221 return (samplerate == player_data->frequency &&
222 ((nr_channels == 1 && player_data->stereo == 0) ||
223 (nr_channels > 1 && player_data->stereo == 1)));
227 remix_player_prepare (RemixEnv *env, RemixBase *base)
229 // remix_player_reset_device (env, base);
234 remix_player_playbuffer (RemixEnv *env, RemixPlayerData *player, RemixPCM *data,
242 length = count * sizeof(RemixCount);
244 for (i = 0; i < count; i++)
246 value = *data++ * (player->max_value);
247 *(player->playbuffer + i) = (PLAYER_PCM) value;
251 ret = mm_sound_pcm_play_write(player->handle,
260 /* An RemixChunkFunc for making noise */
262 remix_player_chunk (RemixEnv *env, RemixChunk *chunk, RemixCount offset,
263 RemixCount count, int channelname, void *data)
265 RemixPlayerData *player = (RemixPlayerData *)data;
266 RemixCount remaining = count, written = 0, n, playcount;
269 while (remaining > 0)
271 playcount = MIN (remaining, player->buffer_size);
273 d = &chunk->data[offset];
274 n = remix_player_playbuffer (env, player, d, playcount);
279 n /= sizeof (PLAYER_PCM);
290 remix_player_process (RemixEnv *env, RemixBase *base, RemixCount count,
291 RemixStream *input, RemixStream *output)
293 RemixCount remaining = count, processed = 0, n, nn, nr_channels;
294 RemixPlayerData *player_data = remix_base_get_instance_data(env, base);
296 if ((!player_data->snd_on) || (!player_data->tch_snd_on)) return count;
297 nr_channels = remix_stream_nr_channels (env, input);
299 if (nr_channels == 1 && player_data->stereo == 0)
301 return remix_stream_chunkfuncify (env, input, count,
302 remix_player_chunk, player_data);
304 else if (nr_channels == 2 && player_data->stereo == 1)
306 while (remaining > 0)
308 n = MIN (remaining, (player_data->buffer_size / 2) );
309 n = remix_stream_interleave_2 (env, input,
310 REMIX_CHANNEL_LEFT, REMIX_CHANNEL_RIGHT,
311 player_data->databuffer, n);
313 nn = remix_player_playbuffer (env, player_data,
314 player_data->databuffer, nn);
323 ERR("[remix_player_process] unsupported stream/output channel\n");
324 ERR ("combination %ld / %d\n", nr_channels, player_data->stereo ? 2 : 1);
330 remix_player_length (RemixEnv *env, RemixBase *base)
332 return REMIX_COUNT_INFINITE;
336 remix_player_seek (RemixEnv *env, RemixBase *base, RemixCount count)
342 _remix_mm_handle_close(void *data)
344 RemixPlayerData *player_data = data;
346 if (!player_data) return ECORE_CALLBACK_CANCEL;
347 if (!player_data->handle) return ECORE_CALLBACK_CANCEL;
349 if (mm_sound_pcm_play_stop(player_data->handle) < 0)
351 player_data->handle = NULL;
352 player_data->timeout = NULL;
353 return ECORE_CALLBACK_CANCEL;
356 mm_sound_pcm_play_close(player_data->handle);
357 player_data->handle = NULL;
358 player_data->timeout = NULL;
360 return ECORE_CALLBACK_CANCEL;
364 remix_player_flush (RemixEnv *env, RemixBase *base)
366 RemixPlayerData *player_data = remix_base_get_instance_data(env, base);
368 if (!player_data) return 0;
370 if (player_data->timeout) ecore_timer_del(player_data->timeout);
371 player_data->timeout = ecore_timer_add(TIMEOUT_FOR_MM_HANDLER,
372 _remix_mm_handle_close, player_data);
374 if (player_data->handle) return 0;
376 player_data->old_buffer_size = player_data->buffer_size;
377 player_data->buffer_size = mm_sound_pcm_play_open_no_session(&player_data->handle,
378 player_data->frequency,
379 player_data->channel,
382 if (mm_sound_pcm_play_start(player_data->handle) < 0)
384 remix_set_error (env, REMIX_ERROR_SYSTEM);
388 if (player_data->buffer_size < 0)
390 remix_set_error (env, REMIX_ERROR_SYSTEM);
394 if (player_data->old_buffer_size < player_data->buffer_size)
396 if (player_data->playbuffer) free(player_data->playbuffer);
397 player_data->playbuffer = calloc(sizeof(PLAYER_PCM), player_data->buffer_size);
398 if (!player_data->playbuffer)
400 remix_set_error (env, REMIX_ERROR_SYSTEM);
408 static struct _RemixMethods _remix_player_methods = {
410 remix_player_destroy,
412 remix_player_prepare,
413 remix_player_process,
420 remix_player_optimise (RemixEnv *env, RemixBase *base)
422 remix_base_set_methods (env, base, &_remix_player_methods);
426 static struct _RemixMetaText tizen_player_metatext = {
428 "TIZEN Sound Player",
429 "Output the stream into TIZEN System",
430 "Copyright (C) 2011, Samsung Electronics Co., Ltd.",
431 "http://www.samsung.com",
432 REMIX_ONE_AUTHOR ("govi.sm@samsung.com", "prince.dubey@samsung.com"),
435 static struct _RemixPlugin tizen_player_plugin = {
436 &tizen_player_metatext,
438 CD_EMPTY_SET, /* init scheme */
440 CD_EMPTY_SET, /* process scheme */
442 NULL, /* plugin data */
447 remix_load (RemixEnv *env)
449 CDList *plugins = cd_list_new (env);
450 plugins = cd_list_prepend (env, plugins,
451 CD_POINTER(&tizen_player_plugin));