source: roaraudio/plugins/audacious/roar.c @ 1594:fe5886a97f79

Last change on this file since 1594:fe5886a97f79 was 1594:fe5886a97f79, checked in by phi, 15 years ago

duplicate semi-colon, moved opening { to correct place

File size: 8.9 KB
Line 
1//roar.c:
2
3/*
4 *      Copyright (C) Philipp 'ph3-der-loewe' Schafft    - 2008,
5 *                    Daniel Duntemann <dauxx@dauxx.org> - 2009
6 *
7 *  This file is part of the Audacious RoarAudio output plugin a part of RoarAudio,
8 *  a cross-platform sound system for both, home and professional use.
9 *  See README for details.
10 *
11 *  This file is free software; you can redistribute it and/or modify
12 *  it under the terms of the GNU General Public License version 3
13 *  as published by the Free Software Foundation.
14 *
15 *  RoarAudio is distributed in the hope that it will be useful,
16 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
17 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18 *  GNU General Public License for more details.
19 *
20 *  You should have received a copy of the GNU General Public License
21 *  along with this software; see the file COPYING.  If not, write to
22 *  the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
23 *
24 */
25
26#include "all.h"
27gint ctrlsocket_get_session_id(void) {
28
29}
30
31OutputPlugin roar_op = {
32    .description = "RoarAudio Output Plugin",         
33    .init = roar_init,
34    .cleanup = NULL,
35    .about = roar_about,
36    .configure = roar_configure, 
37    .get_volume = roar_get_volume,
38    .set_volume = roar_set_volume,
39    .open_audio = roar_open, 
40    .write_audio = roar_write,
41    .close_audio = roar_close,
42    .flush = roar_flush,
43    .pause = roar_pause,
44    .buffer_free = roar_free,
45    .buffer_playing = roar_playing,
46    .output_time = roar_get_output_time, 
47    .written_time = roar_get_written_time,
48    .tell_audio = NULL
49};
50
51OutputPlugin *roar_oplist[] = { &roar_op, NULL };
52
53
54SIMPLE_OUTPUT_PLUGIN("RoarAudio Audacious Plugin",roar_oplist);
55
56OutputPlugin *get_oplugin_info(void) {
57 return &roar_op;
58}
59
60void roar_init(void) {
61 mcs_handle_t * cfgfile;
62
63 cfgfile = aud_cfg_db_open();
64
65 g_inst.state = 0;
66 g_inst.server = NULL;
67 g_inst.session = ctrlsocket_get_session_id();
68
69 aud_cfg_db_get_string(cfgfile, "ROAR", "server", &g_inst.server);
70
71 aud_cfg_db_get_string(cfgfile, "ROAR", "player_name", &g_inst.cfg.player_name);
72
73 aud_cfg_db_close(cfgfile);
74
75 if ( g_inst.cfg.player_name == NULL )
76  g_inst.cfg.player_name = "Audacious";
77
78 ROAR_DBG("roar_init(*) = (void)");
79}
80
81int roar_playing(void) {
82 return FALSE;
83}
84
85void roar_write(void *ptr, int length) {
86 int r;
87
88 if ( g_inst.pause )
89  return;
90
91 ROAR_DBG("roar_write(ptr=%p, length=%i) = (void)", ptr, length);
92
93 while (length) {
94  if ( (r = write(g_inst.data_fh, ptr, length >= 17640 ? 17640 : length)) != -1 ) {
95   g_inst.written   += r;
96   ptr              += r;
97   length           -= r;
98  } else {
99   return;
100  }
101 }
102}
103
104int roar_open(AFormat fmt, int rate, int nch) {
105 int codec = ROAR_CODEC_DEFAULT;
106 int bits;
107
108 if ( !(g_inst.state & STATE_CONNECTED) ) {
109  if ( roar_simple_connect(&(g_inst.con), g_inst.server, g_inst.cfg.player_name) == -1 ) {
110   return FALSE;
111  }
112  g_inst.state |= STATE_CONNECTED;
113 }
114
115  bits = 16;
116  switch (fmt) {
117   case FMT_S8:
118     bits = 8;
119     codec = ROAR_CODEC_DEFAULT;
120    break;
121   case FMT_U8:
122     bits = 8;
123     codec = ROAR_CODEC_PCM_U_LE; // _LE, _BE, _PDP,... all the same for 8 bit output
124    break;
125   case FMT_U16_LE:
126     codec = ROAR_CODEC_PCM_U_LE;
127    break;
128   case FMT_U16_BE:
129     codec = ROAR_CODEC_PCM_U_BE;
130    break;
131   case FMT_U16_NE:
132#if BYTE_ORDER == BIG_ENDIAN
133     codec = ROAR_CODEC_PCM_U_BE;
134#elif BYTE_ORDER == LITTLE_ENDIAN
135     codec = ROAR_CODEC_PCM_U_LE;
136#else
137     codec = ROAR_CODEC_PCM_U_PDP;
138#endif
139    break;
140   case FMT_S16_LE:
141     codec = ROAR_CODEC_PCM_S_LE;
142    break;
143   case FMT_S16_BE:
144     codec = ROAR_CODEC_PCM_S_BE;
145    break;
146   case FMT_S16_NE:
147     codec = ROAR_CODEC_DEFAULT;
148    break;
149    case FMT_U24_LE: /* stored in lower 3 bytes of 32-bit value, highest byte must be 0 */
150     codec = ROAR_CODEC_PCM_U_LE;
151     bits = 24;
152     break;
153    case FMT_U24_BE:
154     codec = ROAR_CODEC_PCM_U_BE;
155     bits = 24;
156     break;
157    case FMT_U24_NE:
158     bits = 24;
159#if BYTE_ORDER == BIG_ENDIAN
160     codec = ROAR_CODEC_PCM_U_BE;
161#elif BYTE_ORDER == LITTLE_ENDIAN
162     codec = ROAR_CODEC_PCM_U_LE;
163#else
164     codec = ROAR_CODEC_PCM_U_PDP;
165#endif     
166     break;
167    case FMT_S24_LE:
168     bits = 24;
169     codec = ROAR_CODEC_PCM_S_LE;
170     break;
171    case FMT_S24_BE:
172     bits = 24;
173     codec = ROAR_CODEC_PCM_S_BE;
174     break;
175    case FMT_S24_NE:
176     codec = ROAR_CODEC_DEFAULT;
177     bits= 24;
178     break;
179    case FMT_U32_LE: /* highest byte must be 0 */
180     codec = ROAR_CODEC_PCM_U_LE;
181     bits = 32;
182     break;
183    case FMT_U32_BE:
184     codec = ROAR_CODEC_PCM_U_BE;
185     bits = 32;
186     break;
187    case FMT_U32_NE:
188     bits = 32;
189#if BYTE_ORDER == BIG_ENDIAN
190     codec = ROAR_CODEC_PCM_U_BE;
191#elif BYTE_ORDER == LITTLE_ENDIAN
192     codec = ROAR_CODEC_PCM_U_LE;
193#else
194     codec = ROAR_CODEC_PCM_U_PDP;
195#endif     
196     break;
197    case FMT_S32_LE:
198     bits = 32;
199     codec = ROAR_CODEC_PCM_S_LE;
200     break;
201    case FMT_S32_BE:
202     bits = 32;
203     codec = ROAR_CODEC_PCM_S_BE;
204     break;
205    case FMT_S32_NE:
206     codec = ROAR_CODEC_DEFAULT;
207     bits= 32;
208     break;
209    case FMT_FLOAT:
210    ROAR_DBG("FMT_FLOAT");
211    break;
212    case FMT_FIXED32:
213     ROAR_DBG("FMT_FIXED32");
214     break;
215     
216 }
217ROAR_DBG("fmt %i",fmt);
218 g_inst.bps       = nch * rate * bits / 8;
219
220 roar_close();
221
222 if ( (g_inst.data_fh = roar_simple_new_stream_obj(&(g_inst.con), &(g_inst.stream),
223                              rate, nch, bits, codec, ROAR_DIR_PLAY)) == -1) {
224  roar_disconnect(&(g_inst.con));
225  g_inst.state |= STATE_CONNECTED;
226  g_inst.state -= STATE_CONNECTED;
227  if ( !(g_inst.state & STATE_NORECONNECT) ) {
228   g_inst.state |= STATE_NORECONNECT;
229   usleep(100000);
230   return roar_open(fmt, rate, nch);
231  } else {
232   g_inst.state -= STATE_NORECONNECT;
233   return FALSE;
234  }
235 }
236 g_inst.state |= STATE_PLAYING;
237
238 g_inst.written = 0;
239 g_inst.pause   = 0;
240
241 roar_update_metadata();
242
243 return TRUE;
244}
245
246void roar_close(void) {
247 if ( g_inst.data_fh != -1 )
248  close(g_inst.data_fh);
249 g_inst.data_fh = -1;
250 g_inst.state |= STATE_PLAYING;
251 g_inst.state -= STATE_PLAYING;
252 g_inst.written = 0;
253 ROAR_DBG("roar_close(void) = (void)");
254}
255
256void roar_pause(short p) {
257 g_inst.pause = p;
258}
259
260int roar_free(void) {
261 if ( g_inst.pause )
262  return 0;
263 else
264  return 1000000; // ???
265}
266
267void roar_flush(int time) {
268 gint64 r = time;
269
270 r *= g_inst.bps;
271 r /= 1000;
272
273 g_inst.written = r;
274}
275
276int roar_get_output_time(void) {
277 return roar_get_written_time();
278}
279
280int roar_get_written_time(void) {
281 gint64 r;
282
283 if ( !g_inst.bps ) {
284  ROAR_DBG("roar_get_written_time(void) = 0");
285  return 0;
286 }
287
288 r  = g_inst.written;
289 r *= 1000; // sec -> msec
290 r /= g_inst.bps;
291 ROAR_DBG("roar_get_written_time(void) = %lu", r);
292
293 return r;
294}
295
296
297// META DATA:
298
299int roar_update_metadata(void) {
300 struct roar_meta   meta;
301 char empty = 0;
302 char * info;
303 int pos;
304
305 pos     = audacious_remote_get_playlist_pos(g_inst.session);
306
307 meta.value = &empty;
308 meta.key[0] = 0;
309 meta.type = ROAR_META_TYPE_NONE;
310
311 roar_stream_meta_set(&(g_inst.con), &(g_inst.stream), ROAR_META_MODE_CLEAR, &meta);
312
313 info = audacious_remote_get_playlist_file(g_inst.session, pos);
314
315 if ( info ) {
316  if ( strncmp(info, "http:", 5) == 0 )
317   meta.type = ROAR_META_TYPE_FILEURL;
318  else
319   meta.type = ROAR_META_TYPE_FILENAME;
320
321  meta.value = info;
322  ROAR_DBG("roar_update_metadata(*): setting meta data: type=%i, strlen(value)=%i", meta.type, strlen(info));
323  roar_stream_meta_set(&(g_inst.con), &(g_inst.stream), ROAR_META_MODE_SET, &meta);
324
325  free(info);
326 }
327
328 info = audacious_remote_get_playlist_title(g_inst.session, pos);
329 if ( info ) {
330  meta.type = ROAR_META_TYPE_TITLE;
331
332  meta.value = info;
333  roar_stream_meta_set(&(g_inst.con), &(g_inst.stream), ROAR_META_MODE_SET, &meta);
334
335  free(info);
336 }
337
338 meta.value = &empty;
339 meta.type = ROAR_META_TYPE_NONE;
340 roar_stream_meta_set(&(g_inst.con), &(g_inst.stream), ROAR_META_MODE_FINALIZE, &meta);
341
342 return 0;
343}
344
345int roar_chk_metadata(void) {
346 static int    old_pos = -1;
347 static char * old_title = "NEW TITLE";
348 int pos;
349 char * title;
350 int need_update = 0;
351
352 pos     = audacious_remote_get_playlist_pos(g_inst.session);
353
354 if ( pos != old_pos ) {
355  old_pos = pos;
356  need_update = 1;
357 } else {
358  title = audacious_remote_get_playlist_title(g_inst.session, pos);
359
360  if ( strcmp(title, old_title) ) {
361   free(old_title);
362   old_title = title;
363   need_update = 1;
364  } else {
365   free(title);
366  }
367 }
368
369 if ( need_update ) {
370  roar_update_metadata();
371 }
372
373 return 0;
374}
375
376// MIXER:
377
378
379void roar_get_volume(int *l, int *r) {
380 int channels;
381 struct roar_mixer_settings mixer;
382
383 if ( !(g_inst.state & STATE_CONNECTED) )
384  return;
385
386 if ( roar_get_vol(&(g_inst.con), g_inst.stream.id, &mixer, &channels) == -1 ) {
387  *l = *r = 100;
388  return;
389 }
390
391 if ( channels == 1 ) {
392  *l = *r = mixer.mixer[0]/655.35;
393 } else {
394  *l = mixer.mixer[0]/655.35;
395  *r = mixer.mixer[1]/655.35;
396 }
397}
398
399void roar_set_volume(int l, int r) {
400 struct roar_mixer_settings mixer;
401
402 if ( !(g_inst.state & STATE_CONNECTED) )
403  return;
404
405 mixer.mixer[0] = l * 655.35;
406 mixer.mixer[1] = r * 655.35;
407
408 roar_set_vol(&(g_inst.con), g_inst.stream.id, &mixer, 2);
409}
410
411//ll
Note: See TracBrowser for help on using the repository browser.