source: roaraudio/plugins/xmms/roar.c @ 3517:1a3218a3fc5b

Last change on this file since 3517:1a3218a3fc5b was 3517:1a3218a3fc5b, checked in by phi, 14 years ago

updated license headers, FSF moved office

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