Mercurial > audlegacy
annotate Plugins/Input/cdaudio/cdaudio.c @ 1519:1ffc2670d6b0 trunk
[svn] - oh, define an actual SDL version here. :P
author | nenolod |
---|---|
date | Mon, 07 Aug 2006 21:22:11 -0700 |
parents | 705d4c089fce |
children | cc5952675fca |
rev | line source |
---|---|
61 | 1 /* XMMS - Cross-platform multimedia player |
2 * Copyright (C) 1998-2003 Peter Alm, Mikael Alm, Olle Hallnas, | |
3 * Thomas Nilsson and 4Front Technologies | |
4 * Copyright (C) 1999-2003 Haavard Kvaalen <havardk@xmms.org> | |
5 * | |
6 * This program is free software; you can redistribute it and/or modify | |
7 * it under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
1459 | 18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
61 | 19 */ |
20 | |
21 #ifdef HAVE_CONFIG_H | |
22 # include "config.h" | |
23 #endif | |
24 | |
25 #include "cdaudio.h" | |
26 | |
27 #include <glib.h> | |
28 #include <glib/gi18n.h> | |
29 #include <glib/gprintf.h> | |
30 #include <string.h> | |
813
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
31 #include <stdlib.h> |
61 | 32 |
33 #include <unistd.h> | |
34 #include <errno.h> | |
35 #include <fcntl.h> | |
36 #include <sys/ioctl.h> | |
37 #include <sys/stat.h> | |
38 | |
39 #include <libaudacious/configdb.h> | |
40 #include <libaudacious/util.h> | |
41 #include <libaudacious/titlestring.h> | |
42 #include "audacious/output.h" | |
43 | |
44 #ifdef CDROMSTOP | |
45 # define XMMS_STOP CDROMSTOP | |
46 #elif defined CDIOCSTOP | |
47 # define XMMS_STOP CDIOCSTOP | |
48 #else | |
49 # error "No stop ioctl" | |
50 #endif | |
51 | |
52 #ifdef CDIOCPAUSE | |
53 # define XMMS_PAUSE CDIOCPAUSE | |
54 #elif defined CDROMPAUSE | |
55 # define XMMS_PAUSE CDROMPAUSE | |
56 #else | |
57 # error "No pause ioctl" | |
58 #endif | |
59 | |
60 #ifdef CDIOCRESUME | |
61 # define XMMS_RESUME CDIOCRESUME | |
62 #elif defined CDROMRESUME | |
63 # define XMMS_RESUME CDROMRESUME | |
64 #else | |
65 # error "No resume ioctl" | |
66 #endif | |
67 | |
68 /* | |
69 * Distributions should not patch this, but instead use the | |
70 * --with-cdda-device=path and --with-cdda-dir=path configure options. | |
71 */ | |
72 | |
73 #ifndef CDDA_DEVICE | |
74 # ifdef HAVE_SYS_CDIO_H | |
392 | 75 # if defined(__FreeBSD__) && !defined(CDIOCREADAUDIO) |
61 | 76 # define CDDA_DEVICE "/dev/acd0c" |
392 | 77 # elif defined __FreeBSD__ |
78 # define CDDA_DEVICE "/dev/acd0" | |
61 | 79 # elif defined __OpenBSD__ |
80 # define CDDA_DEVICE "/dev/cd0c" | |
81 # else | |
82 # define CDDA_DEVICE "/vol/dev/aliases/cdrom0" | |
83 # endif | |
84 # else | |
85 # define CDDA_DEVICE "/dev/cdrom" | |
86 # endif | |
87 #endif | |
88 | |
89 #ifndef CDDA_DIRECTORY | |
90 # ifdef HAVE_SYS_CDIO_H | |
91 # ifdef __FreeBSD__ | |
92 # define CDDA_DIRECTORY "/cdrom" | |
93 # elif defined __OpenBSD__ | |
94 # define CDDA_DIRECTORY "/cdrom" | |
95 # else | |
96 # define CDDA_DIRECTORY "/cdrom/cdrom0" | |
97 # endif | |
98 # else | |
99 # define CDDA_DIRECTORY "/mnt/cdrom" | |
100 # endif | |
101 #endif | |
102 | |
103 | |
104 | |
105 | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
106 static TitleInput *cdda_get_tuple(cdda_disc_toc_t * toc, int track); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
107 static gchar *get_song_title(TitleInput *tuple); |
61 | 108 static gboolean stop_timeout(gpointer data); |
109 | |
110 static void cdda_init(void); | |
111 static int is_our_file(char *filename); | |
112 static GList *scan_dir(char *dir); | |
113 static void play_file(char *filename); | |
114 static void stop(void); | |
115 static void cdda_pause(short p); | |
116 static void seek(int time); | |
117 static int get_time(void); | |
118 static void get_song_info(char *filename, char **title, int *length); | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
119 static TitleInput *get_song_tuple(char *filename); |
61 | 120 static void get_volume(int *l, int *r); |
121 static void set_volume(int l, int r); | |
122 static void cleanup(void); | |
123 void cdda_fileinfo(char *filename); | |
124 | |
125 InputPlugin cdda_ip = { | |
126 NULL, | |
127 NULL, | |
128 NULL, /* Description */ | |
129 cdda_init, | |
130 NULL, /* about */ | |
131 cdda_configure, | |
132 is_our_file, | |
133 scan_dir, | |
134 play_file, | |
135 stop, | |
136 cdda_pause, | |
137 seek, | |
138 NULL, /* set_eq */ | |
139 get_time, | |
140 get_volume, | |
141 set_volume, | |
142 cleanup, | |
143 NULL, /* obsolete */ | |
144 NULL, /* add_vis_pcm */ | |
145 NULL, /* set_info, filled in by xmms */ | |
146 NULL, /* set_info_text, filled in by xmms */ | |
147 get_song_info, | |
148 NULL, /* cdda_fileinfo, *//* file_info_box */ | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
149 NULL, /* output plugin handle */ |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
150 get_song_tuple |
61 | 151 }; |
152 | |
153 CDDAConfig cdda_cfg; | |
154 | |
155 static struct { | |
156 struct driveinfo drive; | |
157 cdda_disc_toc_t cd_toc; | |
158 int track; | |
159 int fd; | |
160 gboolean playing; | |
161 } cdda_playing; | |
162 | |
163 static struct { | |
164 GThread *thread; | |
165 gboolean audio_error, eof; | |
166 int seek; | |
167 | |
168 } dae_data; | |
169 | |
170 static gboolean is_paused; | |
171 static int pause_time; | |
172 | |
173 struct timeout { | |
174 int id; | |
175 char *device; | |
176 }; | |
177 | |
178 static GList *timeout_list; | |
179 | |
180 /* Time to delay stop command in 1/10 second */ | |
181 #define STOP_DELAY 20 | |
182 | |
183 InputPlugin * | |
184 get_iplugin_info(void) | |
185 { | |
186 cdda_ip.description = g_strdup_printf(_("CD Audio Plugin")); | |
187 return &cdda_ip; | |
188 } | |
189 | |
190 | |
191 | |
192 #ifdef BEEP_CDROM_SOLARIS | |
193 /* | |
194 * Lowlevel cdrom access, Solaris style (Solaris, Linux) | |
195 */ | |
196 | |
197 static void | |
198 play_ioctl(struct cdda_msf *start, struct cdda_msf *end) | |
199 { | |
200 struct cdrom_msf msf; | |
201 | |
202 msf.cdmsf_min0 = start->minute; | |
203 msf.cdmsf_sec0 = start->second; | |
204 msf.cdmsf_frame0 = start->frame; | |
205 msf.cdmsf_min1 = end->minute; | |
206 msf.cdmsf_sec1 = end->second; | |
207 msf.cdmsf_frame1 = end->frame; | |
208 ioctl(cdda_playing.fd, CDROMPLAYMSF, &msf); | |
209 } | |
210 | |
211 static int | |
212 get_current_frame(void) | |
213 { | |
214 struct cdrom_subchnl subchnl; | |
215 | |
216 subchnl.cdsc_format = CDROM_MSF; | |
217 if (ioctl(cdda_playing.fd, CDROMSUBCHNL, &subchnl) < 0) | |
218 return -1; | |
219 | |
220 switch (subchnl.cdsc_audiostatus) { | |
221 case CDROM_AUDIO_COMPLETED: | |
222 case CDROM_AUDIO_ERROR: | |
223 return -1; | |
224 } | |
225 | |
226 return (LBA(subchnl.cdsc_absaddr.msf)); | |
227 } | |
228 | |
229 #if !defined(CDROMVOLREAD) | |
230 static int volume_left = 100, volume_right = 100; | |
231 #endif | |
232 | |
233 static void | |
234 drive_get_volume(int *l, int *r) | |
235 { | |
236 #if defined(CDROMVOLREAD) | |
237 struct cdrom_volctrl vol; | |
238 | |
239 if (cdda_playing.fd != -1 && !ioctl(cdda_playing.fd, CDROMVOLREAD, &vol)) { | |
240 *l = (100 * vol.channel0) / 255; | |
241 *r = (100 * vol.channel1) / 255; | |
242 } | |
243 #if 0 | |
244 else if (cdda_playing.fd != -1) | |
245 g_message("CDROMVOLREAD failed"); | |
246 #endif | |
247 #else | |
248 *l = volume_left; | |
249 *r = volume_right; | |
250 #endif | |
251 } | |
252 | |
253 static void | |
254 drive_set_volume(int l, int r) | |
255 { | |
256 struct cdrom_volctrl vol; | |
257 | |
258 if (cdda_playing.fd != -1) { | |
259 vol.channel0 = vol.channel2 = (l * 255) / 100; | |
260 vol.channel1 = vol.channel3 = (r * 255) / 100; | |
261 ioctl(cdda_playing.fd, CDROMVOLCTRL, &vol); | |
262 } | |
263 #if !defined(CDROMVOLREAD) | |
264 volume_left = l; | |
265 volume_right = r; | |
266 #endif | |
267 } | |
268 | |
269 #ifdef CDROMREADAUDIO | |
270 int | |
271 read_audio_data(int fd, int pos, int num, void *buf) | |
272 { | |
273 struct cdrom_read_audio cdra; | |
274 | |
275 #if 1 | |
276 cdra.addr.lba = pos - CDDA_MSF_OFFSET; | |
277 cdra.addr_format = CDROM_LBA; | |
278 #else | |
279 cdra.addr.msf.minute = pos / (60 * 75); | |
280 cdra.addr.msf.second = (pos / 75) % 60; | |
281 cdra.addr.msf.frame = pos % 75; | |
282 cdra.addr_format = CDROM_MSF; | |
283 #endif | |
284 | |
285 cdra.nframes = num; | |
286 cdra.buf = buf; | |
287 | |
288 if (ioctl(fd, CDROMREADAUDIO, &cdra) < 0) | |
289 return -errno; | |
290 | |
291 return cdra.nframes; | |
292 } | |
293 #endif /* CDROMREADAUDIO */ | |
294 | |
295 #if defined(CDROMCDDA) | |
296 int | |
297 read_audio_data(int fd, int pos, int num, void *buf) | |
298 { | |
299 struct cdrom_cdda cdra; | |
300 | |
301 cdra.cdda_addr = pos - CDDA_MSF_OFFSET; | |
302 cdra.cdda_length = num; | |
303 cdra.cdda_data = buf; | |
304 cdra.cdda_subcode = CDROM_DA_NO_SUBCODE; | |
305 if (ioctl(fd, CDROMCDDA, &cdra) < 0) | |
306 return -errno; | |
307 | |
308 return cdra.cdda_length; | |
309 } | |
310 #endif | |
311 | |
312 static gboolean | |
313 cdda_get_toc_lowlevel(int fd, cdda_disc_toc_t * info) | |
314 { | |
315 struct cdrom_tochdr tochdr; | |
316 struct cdrom_tocentry tocentry; | |
317 int i; | |
318 | |
319 | |
320 | |
321 if (ioctl(fd, CDROMREADTOCHDR, &tochdr)) | |
322 return FALSE; | |
323 | |
324 for (i = tochdr.cdth_trk0; i <= tochdr.cdth_trk1; i++) { | |
325 tocentry.cdte_format = CDROM_MSF; | |
326 tocentry.cdte_track = i; | |
327 if (ioctl(fd, CDROMREADTOCENTRY, &tocentry)) | |
328 return FALSE; | |
329 info->track[i].minute = tocentry.cdte_addr.msf.minute; | |
330 info->track[i].second = tocentry.cdte_addr.msf.second; | |
331 info->track[i].frame = tocentry.cdte_addr.msf.frame; | |
332 info->track[i].flags.data_track = | |
333 tocentry.cdte_ctrl == CDROM_DATA_TRACK; | |
334 | |
335 } | |
336 | |
337 /* Get the leadout track */ | |
338 tocentry.cdte_track = CDROM_LEADOUT; | |
339 tocentry.cdte_format = CDROM_MSF; | |
340 | |
341 if (ioctl(fd, CDROMREADTOCENTRY, &tocentry)) | |
342 return FALSE; | |
343 info->leadout.minute = tocentry.cdte_addr.msf.minute; | |
344 info->leadout.second = tocentry.cdte_addr.msf.second; | |
345 info->leadout.frame = tocentry.cdte_addr.msf.frame; | |
346 | |
347 info->first_track = tochdr.cdth_trk0; | |
348 info->last_track = tochdr.cdth_trk1; | |
349 | |
350 return TRUE; | |
351 } | |
352 | |
353 #endif | |
354 | |
355 #ifdef BEEP_CDROM_BSD | |
356 /* | |
357 * Lowlevel cdrom access, BSD style (FreeBSD, OpenBSD, NetBSD, Darwin) | |
358 */ | |
359 | |
360 static void | |
361 play_ioctl(struct cdda_msf *start, struct cdda_msf *end) | |
362 { | |
363 struct ioc_play_msf msf; | |
364 | |
365 msf.start_m = start->minute; | |
366 msf.start_s = start->second; | |
367 msf.start_f = start->frame; | |
368 msf.end_m = end->minute; | |
369 msf.end_s = end->second; | |
370 msf.end_f = end->frame; | |
371 ioctl(cdda_playing.fd, CDIOCPLAYMSF, &msf); | |
372 } | |
373 | |
374 static int | |
375 get_current_frame(void) | |
376 { | |
377 struct ioc_read_subchannel subchnl; | |
378 struct cd_sub_channel_info subinfo; | |
379 subchnl.address_format = CD_MSF_FORMAT; | |
380 subchnl.data_format = CD_CURRENT_POSITION; | |
381 subchnl.track = 0; | |
382 subchnl.data_len = sizeof(subinfo); | |
383 subchnl.data = &subinfo; | |
384 if (ioctl(cdda_playing.fd, CDIOCREADSUBCHANNEL, &subchnl) < 0) | |
385 return -1; | |
386 | |
387 #ifdef BEEP_CDROM_BSD_DARWIN | |
388 return ((subchnl.data->what.position.absaddr[1] * 60 | |
389 subchnl.data->what.position.absaddr[2]) * 75 + | |
390 subchnl.data->what.position.absaddr[3]); | |
391 #else | |
392 return (LBA(subchnl.data->what.position.absaddr.msf)); | |
393 #endif | |
394 } | |
395 | |
396 static void | |
397 drive_get_volume(int *l, int *r) | |
398 { | |
399 struct ioc_vol vol; | |
400 | |
401 if (cdda_playing.fd != -1) { | |
402 ioctl(cdda_playing.fd, CDIOCGETVOL, &vol); | |
403 *l = (100 * vol.vol[0]) / 255; | |
404 *r = (100 * vol.vol[1]) / 255; | |
405 } | |
406 } | |
407 | |
408 static void | |
409 drive_set_volume(int l, int r) | |
410 { | |
411 struct ioc_vol vol; | |
412 | |
413 if (cdda_playing.fd != -1) { | |
414 vol.vol[0] = vol.vol[2] = (l * 255) / 100; | |
415 vol.vol[1] = vol.vol[3] = (r * 255) / 100; | |
416 ioctl(cdda_playing.fd, CDIOCSETVOL, &vol); | |
417 } | |
418 } | |
419 | |
392 | 420 #if defined(__FreeBSD__) && !defined(CDIOCREADAUDIO) |
421 int | |
422 read_audio_data(int fd, int pos, int num, void *buf) | |
423 { | |
424 int bs = CD_FRAMESIZE_RAW; | |
425 | |
426 if (ioctl(fd, CDRIOCSETBLOCKSIZE, &bs) == -1) | |
427 return -1; | |
428 if (pread(fd, buf, num * bs, (pos - 150) * bs) != num * bs) | |
429 return -1; | |
430 | |
431 return num; | |
432 } | |
433 #endif | |
61 | 434 |
435 #if defined(CDIOCREADAUDIO) | |
436 int | |
437 read_audio_data(int fd, int pos, int num, void *buf) | |
438 { | |
439 struct ioc_read_audio cdra; | |
440 | |
441 cdra.address.lba = pos - CDDA_MSF_OFFSET; | |
442 cdra.address_format = CD_LBA_FORMAT; | |
443 cdra.nframes = num; | |
444 cdra.buffer = buf; | |
445 | |
446 if (ioctl(fd, CDIOCREADAUDIO, &cdra) < 0) | |
447 return -errno; | |
448 | |
449 return cdra.nframes; | |
450 } | |
451 #endif /* CDIOCREADAUDIO */ | |
452 | |
453 #ifdef BEEP_CDROM_BSD_NETBSD /* NetBSD, OpenBSD */ | |
454 | |
455 static gboolean | |
456 cdda_get_toc_lowlevel(int fd, cdda_disc_toc_t * info) | |
457 { | |
458 struct ioc_toc_header tochdr; | |
459 struct ioc_read_toc_entry tocentry; | |
460 struct cd_toc_entry tocentrydata; | |
461 int i; | |
462 | |
463 if (ioctl(fd, CDIOREADTOCHEADER, &tochdr)) | |
464 return FALSE; | |
465 | |
466 for (i = tochdr.starting_track; i <= tochdr.ending_track; i++) { | |
467 tocentry.address_format = CD_MSF_FORMAT; | |
468 | |
469 tocentry.starting_track = i; | |
470 tocentry.data = &tocentrydata; | |
471 tocentry.data_len = sizeof(tocentrydata); | |
472 if (ioctl(fd, CDIOREADTOCENTRYS, &tocentry)) | |
473 return FALSE; | |
474 info->track[i].minute = tocentry.data->addr.msf.minute; | |
475 info->track[i].second = tocentry.data->addr.msf.second; | |
476 info->track[i].frame = tocentry.data->addr.msf.frame; | |
477 info->track[i].flags.data_track = (tocentry.data->control & 4) == 4; | |
478 } | |
479 | |
480 /* Get the leadout track */ | |
481 tocentry.address_format = CD_MSF_FORMAT; | |
482 | |
483 tocentry.starting_track = 0xAA; | |
484 tocentry.data = &tocentrydata; | |
485 tocentry.data_len = sizeof(tocentrydata); | |
486 if (ioctl(fd, CDIOREADTOCENTRYS, &tocentry)) | |
487 return FALSE; | |
488 info->leadout.minute = tocentry.data->addr.msf.minute; | |
489 info->leadout.second = tocentry.data->addr.msf.second; | |
490 info->leadout.frame = tocentry.data->addr.msf.frame; | |
491 | |
492 info->first_track = tochdr.starting_track; | |
493 info->last_track = tochdr.ending_track; | |
494 | |
495 return TRUE; | |
496 } | |
497 | |
498 #elif defined(BEEP_CDROM_BSD_DARWIN) | |
499 | |
500 static gboolean | |
501 cdda_get_toc_lowlevel(int fd, cdda_disc_toc_t * info) | |
502 { | |
503 struct ioc_toc_header tochdr; | |
504 struct ioc_read_toc_entry tocentry; | |
505 int i; | |
506 | |
507 if (ioctl(fd, CDIOREADTOCHEADER, &tochdr)) | |
508 return FALSE; | |
509 | |
510 for (i = tochdr.starting_track; i <= tochdr.ending_track; i++) { | |
511 tocentry.address_format = CD_MSF_FORMAT; | |
512 | |
513 tocentry.starting_track = i; | |
514 if (ioctl(fd, CDIOREADTOCENTRYS, &tocentry)) | |
515 return FALSE; | |
516 info->track[i].minute = tocentry.data->addr[1]; | |
517 info->track[i].second = tocentry.data->addr[2]; | |
518 info->track[i].frame = tocentry.data->addr[3]; | |
519 info->track[i].flags.data_track = (tocentry.data->control & 4) == 4; | |
520 } | |
521 | |
522 /* Get the leadout track */ | |
523 tocentry.address_format = CD_MSF_FORMAT; | |
524 | |
525 tocentry.starting_track = 0xAA; | |
526 if (ioctl(fd, CDIOREADTOCENTRYS, &tocentry)) | |
527 return FALSE; | |
528 info->leadout.minute = tocentry.data->addr[1]; | |
529 info->leadout.second = tocentry.data->addr[2]; | |
530 info->leadout.frame = tocentry.data->addr[3]; | |
531 | |
532 return TRUE; | |
533 } | |
534 | |
535 #else /* FreeBSD */ | |
536 | |
537 static gboolean | |
538 cdda_get_toc_lowlevel(int fd, cdda_disc_toc_t * info) | |
539 { | |
540 struct ioc_toc_header tochdr; | |
541 struct ioc_read_toc_single_entry tocentry; | |
542 int i; | |
543 | |
544 if (ioctl(fd, CDIOREADTOCHEADER, &tochdr)) | |
545 return FALSE; | |
546 | |
547 for (i = tochdr.starting_track; i <= tochdr.ending_track; i++) { | |
548 tocentry.address_format = CD_MSF_FORMAT; | |
549 | |
550 tocentry.track = i; | |
551 if (ioctl(fd, CDIOREADTOCENTRY, &tocentry)) | |
552 return FALSE; | |
553 info->track[i].minute = tocentry.entry.addr.msf.minute; | |
554 info->track[i].second = tocentry.entry.addr.msf.second; | |
555 info->track[i].frame = tocentry.entry.addr.msf.frame; | |
556 info->track[i].flags.data_track = (tocentry.entry.control & 4) == 4; | |
557 } | |
558 | |
559 /* Get the leadout track */ | |
560 tocentry.address_format = CD_MSF_FORMAT; | |
561 | |
562 tocentry.track = 0xAA; | |
563 if (ioctl(fd, CDIOREADTOCENTRY, &tocentry)) | |
564 return FALSE; | |
565 info->leadout.minute = tocentry.entry.addr.msf.minute; | |
566 info->leadout.second = tocentry.entry.addr.msf.second; | |
567 info->leadout.frame = tocentry.entry.addr.msf.frame; | |
568 | |
569 info->first_track = tochdr.starting_track; | |
570 info->last_track = tochdr.ending_track; | |
571 | |
572 return TRUE; | |
573 } | |
574 #endif | |
575 | |
576 #endif | |
577 | |
578 | |
579 | |
580 | |
581 | |
582 | |
583 | |
584 | |
585 | |
586 | |
587 | |
588 extern gboolean | |
589 is_mounted(const char *device_name) | |
590 { | |
591 #if defined(HAVE_MNTENT_H) || defined(HAVE_GETMNTINFO) | |
592 char devname[256]; | |
593 struct stat st; | |
594 #if defined(HAVE_MNTENT_H) | |
595 FILE *mounts; | |
596 struct mntent *mnt; | |
597 #elif defined(HAVE_GETMNTINFO) | |
598 struct statfs *fsp; | |
599 int entries; | |
600 #endif | |
601 | |
602 if (lstat(device_name, &st) < 0) | |
603 return -1; | |
604 | |
605 if (S_ISLNK(st.st_mode)) | |
606 readlink(device_name, devname, 256); | |
607 else | |
608 strncpy(devname, device_name, 256); | |
609 | |
610 #if defined(HAVE_MNTENT_H) | |
611 if ((mounts = setmntent(MOUNTED, "r")) == NULL) | |
612 return TRUE; | |
613 | |
614 while ((mnt = getmntent(mounts)) != NULL) { | |
615 if (strcmp(mnt->mnt_fsname, devname) == 0) { | |
616 endmntent(mounts); | |
617 return TRUE; | |
618 } | |
619 } | |
620 endmntent(mounts); | |
621 #elif defined(HAVE_GETMNTINFO) | |
622 entries = getmntinfo(&fsp, MNT_NOWAIT); | |
623 if (entries < 0) | |
419
2291caab50ca
[svn] FreeBSD 6.x/7.x newata cdrio fixes. (I now have an ia32 box to work with ;))
nenolod
parents:
392
diff
changeset
|
624 return FALSE; |
61 | 625 |
626 while (entries-- > 0) { | |
627 if (!strcmp(fsp->f_mntfromname, devname)) | |
628 return TRUE; | |
629 fsp++; | |
630 } | |
631 #endif | |
632 #endif | |
633 return FALSE; | |
634 } | |
635 | |
636 | |
637 gboolean | |
638 cdda_get_toc(cdda_disc_toc_t * info, const char *device) | |
639 { | |
640 gboolean retv = FALSE; | |
641 int fd; | |
642 | |
643 if (is_mounted(device)) | |
644 return FALSE; | |
645 | |
646 if ((fd = open(device, CDOPENFLAGS)) == -1) | |
647 return FALSE; | |
648 | |
649 memset(info, 0, sizeof(cdda_disc_toc_t)); | |
650 | |
651 retv = cdda_get_toc_lowlevel(fd, info); | |
652 close(fd); | |
653 | |
654 return retv; | |
655 } | |
656 | |
657 static void | |
658 cdda_init(void) | |
659 { | |
660 ConfigDb *db; | |
661 struct driveinfo *drive = g_malloc0(sizeof(struct driveinfo)); | |
662 int ndrives = 1, i; | |
663 | |
664 cdda_playing.fd = -1; | |
665 memset(&cdda_cfg, 0, sizeof(CDDAConfig)); | |
666 | |
667 #ifdef HAVE_OSS | |
668 drive->mixer = CDDA_MIXER_OSS; | |
669 drive->oss_mixer = SOUND_MIXER_CD; | |
670 #endif | |
671 | |
672 db = bmp_cfg_db_open(); | |
673 | |
674 /* These names are used for backwards compatibility */ | |
675 bmp_cfg_db_get_string(db, "CDDA", "device", &drive->device); | |
676 bmp_cfg_db_get_string(db, "CDDA", "directory", &drive->directory); | |
677 bmp_cfg_db_get_int(db, "CDDA", "mixer", &drive->mixer); | |
678 bmp_cfg_db_get_int(db, "CDDA", "readmode", &drive->dae); | |
679 | |
680 if (!drive->device) | |
681 drive->device = g_strdup(CDDA_DEVICE); | |
682 if (!drive->directory) | |
683 drive->directory = g_strdup(CDDA_DIRECTORY); | |
684 | |
685 cdda_cfg.drives = g_list_append(cdda_cfg.drives, drive); | |
686 | |
687 bmp_cfg_db_get_int(db, "CDDA", "num_drives", &ndrives); | |
688 for (i = 1; i < ndrives; i++) { | |
689 char label[20]; | |
690 drive = g_malloc0(sizeof(struct driveinfo)); | |
691 | |
692 sprintf(label, "device%d", i); | |
693 bmp_cfg_db_get_string(db, "CDDA", label, &drive->device); | |
694 | |
695 sprintf(label, "directory%d", i); | |
696 bmp_cfg_db_get_string(db, "CDDA", label, &drive->directory); | |
697 | |
698 sprintf(label, "mixer%d", i); | |
699 bmp_cfg_db_get_int(db, "CDDA", label, &drive->mixer); | |
700 | |
701 sprintf(label, "readmode%d", i); | |
702 bmp_cfg_db_get_int(db, "CDDA", label, &drive->dae); | |
703 | |
704 cdda_cfg.drives = g_list_append(cdda_cfg.drives, drive); | |
705 } | |
706 bmp_cfg_db_get_bool(db, "CDDA", "title_override", | |
707 &cdda_cfg.title_override); | |
708 bmp_cfg_db_get_string(db, "CDDA", "name_format", &cdda_cfg.name_format); | |
709 bmp_cfg_db_get_bool(db, "CDDA", "use_cddb", &cdda_cfg.use_cddb); | |
710 bmp_cfg_db_get_string(db, "CDDA", "cddb_server", &cdda_cfg.cddb_server); | |
711 #ifdef WITH_CDINDEX | |
712 bmp_cfg_db_get_bool(db, "CDDA", "use_cdin", &cdda_cfg.use_cdin); | |
713 #else | |
714 cdda_cfg.use_cdin = FALSE; | |
715 #endif | |
716 bmp_cfg_db_get_string(db, "CDDA", "cdin_server", &cdda_cfg.cdin_server); | |
717 bmp_cfg_db_close(db); | |
718 | |
719 if (!cdda_cfg.cdin_server) | |
720 cdda_cfg.cdin_server = g_strdup("www.cdindex.org"); | |
721 if (!cdda_cfg.cddb_server) | |
722 cdda_cfg.cddb_server = g_strdup(CDDB_DEFAULT_SERVER); | |
723 if (!cdda_cfg.name_format) | |
724 cdda_cfg.name_format = g_strdup("%p - %t"); | |
725 } | |
726 | |
727 struct driveinfo * | |
728 cdda_find_drive(char *filename) | |
729 { | |
730 GList *node; | |
731 | |
732 // FIXME: Will always return the first drive | |
733 | |
734 for (node = cdda_cfg.drives; node; node = node->next) { | |
735 struct driveinfo *d = node->data; | |
736 if (!strncmp(d->directory, filename, strlen(d->directory))) | |
737 return d; | |
738 } | |
739 | |
740 return NULL; | |
741 | |
742 } | |
743 | |
744 static void | |
745 timeout_destroy(struct timeout *entry) | |
746 { | |
747 g_free(entry->device); | |
748 g_free(entry); | |
749 timeout_list = g_list_remove(timeout_list, entry); | |
750 } | |
751 | |
752 static void | |
753 timeout_remove_for_device(char *device) | |
754 { | |
755 GList *node; | |
756 | |
757 for (node = timeout_list; node; node = node->next) { | |
758 struct timeout *t = node->data; | |
759 | |
760 if (!strcmp(t->device, device)) { | |
761 gtk_timeout_remove(t->id); | |
762 timeout_destroy(t); | |
763 return; | |
764 } | |
765 } | |
766 | |
767 } | |
768 | |
769 static void | |
770 cleanup(void) | |
771 { | |
813
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
772 GList *node; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
773 struct driveinfo *drive; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
774 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
775 g_free(cdda_ip.description); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
776 cdda_ip.description = NULL; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
777 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
778 if (cdda_cfg.drives) { |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
779 for (node = g_list_first(cdda_cfg.drives); node; node = node->next) { |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
780 drive = (struct driveinfo *)node->data; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
781 if (!drive) |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
782 continue; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
783 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
784 if (drive->device) |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
785 free(drive->device); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
786 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
787 if (drive->directory) |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
788 free(drive->directory); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
789 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
790 free(drive); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
791 } |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
792 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
793 g_list_free(cdda_cfg.drives); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
794 cdda_cfg.drives = NULL; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
795 } |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
796 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
797 if (cdda_cfg.name_format) { |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
798 free(cdda_cfg.name_format); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
799 cdda_cfg.name_format = NULL; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
800 } |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
801 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
802 if (cdda_cfg.cddb_server) { |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
803 free(cdda_cfg.cddb_server); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
804 cdda_cfg.cddb_server = NULL; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
805 } |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
806 |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
807 if (cdda_cfg.cdin_server) { |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
808 free(cdda_cfg.cdin_server); |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
809 cdda_cfg.cdin_server = NULL; |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
810 } |
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
419
diff
changeset
|
811 |
61 | 812 while (timeout_list) { |
813 struct timeout *t = timeout_list->data; | |
814 gtk_timeout_remove(t->id); | |
815 stop_timeout(t); | |
816 timeout_destroy(t); | |
817 } | |
818 cddb_quit(); | |
819 } | |
820 | |
821 static int | |
822 is_our_file(char *filename) | |
823 { | |
824 char *ext = ".cda"; | |
825 | |
826 if (cdda_find_drive(filename) == NULL) { | |
827 return FALSE; | |
828 } | |
829 | |
830 if (g_str_has_suffix(filename, ext)) { | |
831 return TRUE; | |
832 } | |
833 return FALSE; | |
834 } | |
835 | |
836 | |
837 static GList * | |
838 scan_dir(char *dir) | |
839 { | |
840 GList *list = NULL; | |
841 int i; | |
842 cdda_disc_toc_t toc; | |
843 struct driveinfo *drive; | |
844 | |
845 if ((drive = cdda_find_drive(dir)) == NULL) | |
846 return NULL; | |
847 | |
848 if (!cdda_get_toc(&toc, drive->device)) | |
849 return NULL; | |
850 | |
851 for (i = toc.last_track; i >= toc.first_track; i--) | |
852 if (!toc.track[i].flags.data_track) { | |
853 list = g_list_prepend(list, g_strdup_printf("Track %02d.cda", i)); | |
854 } | |
855 return list; | |
856 } | |
857 | |
858 guint | |
859 cdda_calculate_track_length(cdda_disc_toc_t * toc, int track) | |
860 { | |
861 if (track == toc->last_track) | |
862 return (LBA(toc->leadout) - LBA(toc->track[track])); | |
863 else | |
864 return (LBA(toc->track[track + 1]) - LBA(toc->track[track])); | |
865 } | |
866 | |
867 static void * | |
868 dae_play_loop(void *arg) | |
869 { | |
870 char *buffer = g_malloc(CD_FRAMESIZE_RAW * CDDA_DAE_FRAMES); | |
871 int pos = LBA(cdda_playing.cd_toc.track[cdda_playing.track]); | |
872 int end, frames; | |
873 | |
874 if (cdda_playing.track == cdda_playing.cd_toc.last_track) | |
875 end = LBA(cdda_playing.cd_toc.leadout); | |
876 else | |
877 end = LBA(cdda_playing.cd_toc.track[cdda_playing.track + 1]); | |
878 | |
879 while (cdda_playing.playing) { | |
880 int left; | |
881 char *data; | |
882 | |
883 if (dae_data.seek != -1) { | |
884 cdda_ip.output->flush(dae_data.seek * 1000); | |
885 pos = LBA(cdda_playing.cd_toc.track[cdda_playing.track]) | |
886 + dae_data.seek * 75; | |
887 dae_data.seek = -1; | |
888 dae_data.eof = FALSE; | |
889 } | |
890 frames = MIN(CDDA_DAE_FRAMES, end - pos); | |
891 if (frames == 0) | |
892 dae_data.eof = TRUE; | |
893 | |
894 if (dae_data.eof) { | |
895 xmms_usleep(30000); | |
896 continue; | |
897 } | |
898 | |
899 frames = read_audio_data(cdda_playing.fd, pos, frames, buffer); | |
900 if (frames <= 0) { | |
901 int err = -frames; | |
902 if (err == EOPNOTSUPP) | |
903 dae_data.eof = TRUE; | |
904 else { | |
905 /* | |
906 * If the read failed, skip ahead to | |
907 * avoid getting stuck on scratches | |
908 * and such. | |
909 */ | |
910 g_message("read_audio_data() failed: %s (%d)", | |
911 strerror(err), err); | |
912 pos += MIN(CDDA_DAE_FRAMES, end - pos); | |
913 } | |
914 continue; | |
915 } | |
916 left = frames * CD_FRAMESIZE_RAW; | |
917 data = buffer; | |
918 while (cdda_playing.playing && left > 0 && dae_data.seek == -1) { | |
919 int cur = MIN(512 * 2 * 2, left); | |
920 cdda_ip.add_vis_pcm(cdda_ip.output->written_time(), | |
921 FMT_S16_LE, 2, cur, data); | |
922 while (cdda_ip.output->buffer_free() < cur && | |
923 cdda_playing.playing && dae_data.seek == -1) | |
924 xmms_usleep(30000); | |
925 if (cdda_playing.playing && dae_data.seek == -1) | |
926 produce_audio(cdda_ip.output->written_time(), FMT_S16_LE, 2, cur, data, &cdda_playing.playing); | |
927 left -= cur; | |
928 data += cur; | |
929 } | |
930 pos += frames; | |
931 } | |
932 | |
933 cdda_ip.output->buffer_free(); | |
934 cdda_ip.output->buffer_free(); | |
935 g_free(buffer); | |
936 | |
937 g_thread_exit(NULL); | |
938 return NULL; | |
939 } | |
940 | |
941 static void | |
942 dae_play(void) | |
943 { | |
944 if (cdda_ip.output->open_audio(FMT_S16_LE, 44100, 2) == 0) { | |
945 dae_data.audio_error = TRUE; | |
946 cdda_playing.playing = FALSE; | |
947 return; | |
948 } | |
949 dae_data.seek = -1; | |
950 dae_data.eof = FALSE; | |
951 dae_data.audio_error = FALSE; | |
952 dae_data.thread = g_thread_create(dae_play_loop, NULL, TRUE, NULL); | |
953 } | |
954 | |
955 static void | |
956 play_file(char *filename) | |
957 { | |
958 char *tmp; | |
959 struct driveinfo *drive; | |
960 int track; | |
961 int track_len; | |
962 | |
963 // g_message(g_strdup_printf("** CD_AUDIO: trying to play file %s",filename)); | |
964 | |
965 if ((drive = cdda_find_drive(filename)) == NULL) { | |
966 // g_message("** CD_AUDIO: find drive check failed"); | |
967 return; | |
968 } | |
969 if (is_mounted(drive->device)) { | |
970 // g_message("** CD_AUDIO: drive is mounted"); | |
971 return; | |
972 } | |
973 tmp = strrchr(filename, '/'); | |
974 if (tmp) | |
975 tmp++; | |
976 else | |
977 tmp = filename; | |
978 | |
979 if (!sscanf(tmp, "Track %d.cda", &track)) { | |
980 // g_message("** CD_AUDIO: filename check failed"); | |
981 return; | |
982 } | |
983 | |
984 if (!cdda_get_toc(&cdda_playing.cd_toc, drive->device) || | |
985 cdda_playing.cd_toc.track[track].flags.data_track || | |
986 track < cdda_playing.cd_toc.first_track || | |
987 track > cdda_playing.cd_toc.last_track) { | |
988 // g_message("** CD_AUDIO: toc check failed"); | |
989 return; | |
990 } | |
991 | |
992 if ((cdda_playing.fd = open(drive->device, CDOPENFLAGS)) == -1) { | |
993 // g_message("** CD_AUDIO: device open failed"); | |
994 return; | |
995 } | |
996 track_len = cdda_calculate_track_length(&cdda_playing.cd_toc, track); | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
997 cdda_ip.set_info(get_song_title(cdda_get_tuple(&cdda_playing.cd_toc, track)), |
61 | 998 (track_len * 1000) / 75, 44100 * 2 * 2 * 8, 44100, 2); |
999 | |
1000 memcpy(&cdda_playing.drive, drive, sizeof(struct driveinfo)); | |
1001 #ifndef CDDA_HAS_READAUDIO | |
1002 cdda_playing.drive.dae = FALSE; | |
1003 #endif | |
1004 | |
1005 cdda_playing.track = track; | |
1006 | |
1007 is_paused = FALSE; | |
1008 timeout_remove_for_device(drive->device); | |
1009 | |
1010 cdda_playing.playing = TRUE; | |
1011 if (drive->dae) | |
1012 dae_play(); | |
1013 else | |
1014 seek(0); | |
1015 } | |
1016 | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1017 static TitleInput * |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1018 cdda_get_tuple(cdda_disc_toc_t * toc, int track) |
61 | 1019 { |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1020 G_LOCK_DEFINE_STATIC(tuple); |
61 | 1021 |
1022 static guint32 cached_id; | |
1023 static cdinfo_t cdinfo; | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1024 static gchar *performer, *album_name, *track_name; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1025 TitleInput *tuple = NULL; |
61 | 1026 guint32 disc_id; |
1027 | |
1028 disc_id = cdda_cddb_compute_discid(toc); | |
1029 | |
1030 /* | |
1031 * We want to avoid looking up a album from two threads simultaneously. | |
1032 * This can happen since we are called both from the main-thread and | |
1033 * from the playlist-thread. | |
1034 */ | |
1035 | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1036 G_LOCK(tuple); |
61 | 1037 if (!(disc_id == cached_id && cdinfo.is_valid)) { |
1038 /* | |
1039 * We try to look up the disc again if the info is not | |
1040 * valid. The user might have configured a new server | |
1041 * in the meantime. | |
1042 */ | |
1043 cdda_cdinfo_flush(&cdinfo); | |
1044 cached_id = disc_id; | |
1045 | |
1046 if (!cdda_cdinfo_read_file(disc_id, &cdinfo)) { | |
1047 if (cdda_cfg.use_cddb) | |
1048 cdda_cddb_get_info(toc, &cdinfo); | |
1049 if (cdinfo.is_valid) | |
1050 cdda_cdinfo_write_file(disc_id, &cdinfo); | |
1051 } | |
1052 } | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1053 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1054 tuple = bmp_title_input_new(); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1055 cdda_cdinfo_get(&cdinfo, track, &performer, &album_name, &track_name); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1056 G_UNLOCK(tuple); |
61 | 1057 |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1058 tuple->performer = g_strdup(performer); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1059 tuple->album_name = g_strdup(album_name); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1060 tuple->track_name = g_strdup(track_name); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1061 tuple->track_number = (track); |
1364 | 1062 tuple->file_name = g_strdup(tuple->file_path); |
1063 tuple->file_path = g_strdup_printf(_("CD Audio Track %02u"), track); | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1064 tuple->file_ext = "cda"; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1065 tuple->length = ((cdda_calculate_track_length(toc, track) * 1000) / 75); |
61 | 1066 |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1067 if (!tuple->track_name) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1068 tuple->track_name = g_strdup_printf(_("CD Audio Track %02u"), track); |
1364 | 1069 |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1070 return tuple; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1071 } |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1072 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1073 static gchar * |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1074 get_song_title(TitleInput *tuple) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1075 { |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1076 return xmms_get_titlestring(cdda_cfg.title_override ? |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1077 cdda_cfg.name_format : |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1078 xmms_get_gentitle_format(), tuple); |
61 | 1079 } |
1080 | |
1081 static gboolean | |
1082 stop_timeout(gpointer data) | |
1083 { | |
1084 int fd; | |
1085 struct timeout *to = data; | |
1086 | |
1087 fd = open(to->device, CDOPENFLAGS); | |
1088 if (fd != -1) { | |
1089 ioctl(fd, XMMS_STOP, 0); | |
1090 close(fd); | |
1091 } | |
1092 timeout_destroy(to); | |
1093 return FALSE; | |
1094 } | |
1095 | |
1096 static void | |
1097 stop(void) | |
1098 { | |
1099 struct timeout *to_info; | |
1100 if (cdda_playing.fd < 0) | |
1101 return; | |
1102 | |
1103 cdda_playing.playing = FALSE; | |
1104 | |
1105 if (cdda_playing.drive.dae) { | |
1106 g_thread_join(dae_data.thread); | |
1107 cdda_ip.output->close_audio(); | |
1108 } | |
1109 else | |
1110 ioctl(cdda_playing.fd, XMMS_PAUSE, 0); | |
1111 | |
1112 close(cdda_playing.fd); | |
1113 cdda_playing.fd = -1; | |
1114 | |
1115 if (!cdda_playing.drive.dae) { | |
1116 to_info = g_malloc(sizeof(*to_info)); | |
1117 to_info->device = g_strdup(cdda_playing.drive.device); | |
1118 to_info->id = gtk_timeout_add(STOP_DELAY * 100, stop_timeout, | |
1119 to_info); | |
1120 timeout_list = g_list_prepend(timeout_list, to_info); | |
1121 } | |
1122 } | |
1123 | |
1124 static void | |
1125 cdda_pause(short p) | |
1126 { | |
1127 if (cdda_playing.drive.dae) { | |
1128 cdda_ip.output->pause(p); | |
1129 return; | |
1130 } | |
1131 if (p) { | |
1132 pause_time = get_time(); | |
1133 ioctl(cdda_playing.fd, XMMS_PAUSE, 0); | |
1134 } | |
1135 else { | |
1136 ioctl(cdda_playing.fd, XMMS_RESUME, 0); | |
1137 pause_time = -1; | |
1138 } | |
1139 is_paused = p; | |
1140 } | |
1141 | |
1142 | |
1143 | |
1144 static void | |
1145 seek(int time) | |
1146 { | |
1147 struct cdda_msf *end, start; | |
1148 int track = cdda_playing.track; | |
1149 | |
1150 // g_message("** CD_AUDIO: seeking..."); | |
1151 if (cdda_playing.drive.dae) { | |
1152 dae_data.seek = time; | |
1153 while (dae_data.seek != -1) | |
1154 xmms_usleep(20000); | |
1155 return; | |
1156 } | |
1157 | |
1158 start.minute = (cdda_playing.cd_toc.track[track].minute * 60 + | |
1159 cdda_playing.cd_toc.track[track].second + time) / 60; | |
1160 start.second = (cdda_playing.cd_toc.track[track].second + time) % 60; | |
1161 start.frame = cdda_playing.cd_toc.track[track].frame; | |
1162 if (track == cdda_playing.cd_toc.last_track) | |
1163 end = &cdda_playing.cd_toc.leadout; | |
1164 else | |
1165 end = &cdda_playing.cd_toc.track[track + 1]; | |
1166 | |
1167 play_ioctl(&start, end); | |
1168 | |
1169 if (is_paused) { | |
1170 cdda_pause(TRUE); | |
1171 pause_time = time * 1000; | |
1172 } | |
1173 } | |
1174 | |
1175 static int | |
1176 get_time_analog(void) | |
1177 { | |
1178 int frame, start_frame, length; | |
1179 int track = cdda_playing.track; | |
1180 | |
1181 if (is_paused && pause_time != -1) | |
1182 return pause_time; | |
1183 | |
1184 frame = get_current_frame(); | |
1185 | |
1186 if (frame == -1) | |
1187 return -1; | |
1188 | |
1189 start_frame = LBA(cdda_playing.cd_toc.track[track]); | |
1190 length = cdda_calculate_track_length(&cdda_playing.cd_toc, track); | |
1191 | |
1192 if (frame - start_frame >= length - 20) /* 20 seems to work better */ | |
1193 return -1; | |
1194 | |
1195 return ((frame - start_frame) * 1000) / 75; | |
1196 } | |
1197 | |
1198 static int | |
1199 get_time_dae(void) | |
1200 { | |
1201 if (dae_data.audio_error) | |
1202 return -2; | |
1203 if (!cdda_playing.playing || | |
1204 (dae_data.eof && !cdda_ip.output->buffer_playing())) | |
1205 return -1; | |
1206 return cdda_ip.output->output_time(); | |
1207 } | |
1208 | |
1209 static int | |
1210 get_time(void) | |
1211 { | |
1212 if (cdda_playing.fd == -1) | |
1213 return -1; | |
1214 | |
1215 if (cdda_playing.drive.dae) | |
1216 return get_time_dae(); | |
1217 else | |
1218 return get_time_analog(); | |
1219 } | |
1220 | |
1221 static void | |
1222 get_song_info(char *filename, char **title, int *len) | |
1223 { | |
1224 cdda_disc_toc_t toc; | |
1225 int t; | |
1226 char *tmp; | |
1227 struct driveinfo *drive; | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1228 TitleInput *tuple; |
61 | 1229 |
1230 *title = NULL; | |
1231 *len = -1; | |
1232 | |
1233 // g_message("** CD_AUDIO: getting song info"); | |
1234 | |
1235 if ((drive = cdda_find_drive(filename)) == NULL) | |
1236 return; | |
1237 | |
1238 tmp = strrchr(filename, '/'); | |
1239 if (tmp) | |
1240 tmp++; | |
1241 else | |
1242 tmp = filename; | |
1243 | |
1244 if (!sscanf(tmp, "Track %d.cda", &t)) | |
1245 return; | |
1246 if (!cdda_get_toc(&toc, drive->device)) | |
1247 return; | |
1248 if (t < toc.first_track || t > toc.last_track | |
1249 || toc.track[t].flags.data_track) | |
1250 return; | |
1251 | |
1318
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1252 if ((tuple = cdda_get_tuple(&toc, t)) != NULL) { |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1253 *len = tuple->length; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1254 *title = get_song_title(tuple); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1255 } |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1256 bmp_title_input_free(tuple); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1257 } |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1258 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1259 static TitleInput * |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1260 get_song_tuple(char *filename) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1261 { |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1262 cdda_disc_toc_t toc; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1263 int t; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1264 char *tmp; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1265 struct driveinfo *drive; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1266 TitleInput *tuple = NULL; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1267 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1268 // g_message("** CD_AUDIO: getting song info"); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1269 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1270 if ((drive = cdda_find_drive(filename)) == NULL) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1271 return tuple; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1272 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1273 tmp = strrchr(filename, '/'); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1274 if (tmp) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1275 tmp++; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1276 else |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1277 tmp = filename; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1278 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1279 if (!sscanf(tmp, "Track %d.cda", &t)) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1280 return tuple; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1281 if (!cdda_get_toc(&toc, drive->device)) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1282 return tuple; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1283 if (t < toc.first_track || t > toc.last_track |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1284 || toc.track[t].flags.data_track) |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1285 return tuple; |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1286 |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1287 tuple = cdda_get_tuple(&toc, t); |
a344e9836a4e
[svn] - cdaudio playlist tuple support; needs some fixing
nhjm449
parents:
813
diff
changeset
|
1288 return tuple; |
61 | 1289 } |
1290 | |
1291 #ifdef HAVE_OSS | |
1292 static void | |
1293 oss_get_volume(int *l, int *r, int mixer_line) | |
1294 { | |
1295 int fd, v; | |
1296 | |
1297 fd = open(DEV_MIXER, O_RDONLY); | |
1298 if (fd != -1) { | |
1299 ioctl(fd, MIXER_READ(mixer_line), &v); | |
1300 *r = (v & 0xFF00) >> 8; | |
1301 *l = (v & 0x00FF); | |
1302 close(fd); | |
1303 } | |
1304 } | |
1305 | |
1306 static void | |
1307 oss_set_volume(int l, int r, int mixer_line) | |
1308 { | |
1309 int fd, v; | |
1310 | |
1311 fd = open(DEV_MIXER, O_RDONLY); | |
1312 if (fd != -1) { | |
1313 v = (r << 8) | l; | |
1314 ioctl(fd, MIXER_WRITE(mixer_line), &v); | |
1315 close(fd); | |
1316 } | |
1317 } | |
1318 #else | |
1319 static void | |
1320 oss_get_volume(int *l, int *r, int mixer_line) | |
1321 { | |
1322 } | |
1323 static void | |
1324 oss_set_volume(int l, int r, int mixer_line) | |
1325 { | |
1326 } | |
1327 #endif | |
1328 | |
1329 | |
1330 static void | |
1331 get_volume(int *l, int *r) | |
1332 { | |
1333 if (cdda_playing.drive.dae) | |
1334 cdda_ip.output->get_volume(l, r); | |
1335 else if (cdda_playing.drive.mixer == CDDA_MIXER_OSS) | |
1336 oss_get_volume(l, r, cdda_playing.drive.oss_mixer); | |
1337 else if (cdda_playing.drive.mixer == CDDA_MIXER_DRIVE) | |
1338 drive_get_volume(l, r); | |
1339 } | |
1340 | |
1341 static void | |
1342 set_volume(int l, int r) | |
1343 { | |
1344 if (cdda_playing.drive.dae) | |
1345 cdda_ip.output->set_volume(l, r); | |
1346 else if (cdda_playing.drive.mixer == CDDA_MIXER_OSS) | |
1347 oss_set_volume(l, r, cdda_playing.drive.oss_mixer); | |
1348 else if (cdda_playing.drive.mixer == CDDA_MIXER_DRIVE) | |
1349 drive_set_volume(l, r); | |
1350 } |