source: roaraudio/libroar/cdrom.c @ 1394:b8703c5336a6

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

make cdrom support optional

File size: 6.1 KB
Line 
1//cdrom.c:
2
3/*
4 *      Copyright (C) Philipp 'ph3-der-loewe' Schafft - 2008
5 *
6 *  This file is part of libroar 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 *  libroar 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, 675 Mass Ave, Cambridge, MA 02139, USA.
22 *
23 *  NOTE for everyone want's to change something and send patches:
24 *  read README and HACKING! There a addition information on
25 *  the license of this document you need to read before you send
26 *  any patches.
27 *
28 *  NOTE for uses of non-GPL (LGPL,...) software using libesd, libartsc
29 *  or libpulse*:
30 *  The libs libroaresd, libroararts and libroarpulse link this lib
31 *  and are therefore GPL. Because of this it may be illigal to use
32 *  them with any software that uses libesd, libartsc or libpulse*.
33 */
34
35#include "libroar.h"
36#include <sys/types.h>
37#include <signal.h>
38
39#define ROAR_CDROM_ERROR_NORETURN(format, args...) ROAR_ERR(format, ## args); _exit(3)
40
41#if BYTE_ORDER == BIG_ENDIAN
42#define ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT "--output-raw-big-endian"
43#elif BYTE_ORDER == LITTLE_ENDIAN
44#define ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT "--output-raw-little-endian"
45#endif
46
47#ifdef ROAR_TARGET_WIN32
48#undef ROAR_HAVE_BIN_CDPARANOIA
49#endif
50
51pid_t roar_cdrom_run_cdparanoia (int cdrom, int data, int track, char * pos) {
52#if defined(ROAR_HAVE_BIN_CDPARANOIA) && defined(ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT) && defined(ROAR_HAVE_CDROM)
53 char my_pos[32] = {0};
54 pid_t pid;
55 int fh[2];
56
57 ROAR_DBG("roar_cdrom_run_cdparanoia(cdrom=%i, data=%i, track=%i, pos='%s') = ?", cdrom, data, track, pos);
58
59 if ( cdrom == -1 || data == -1 || (track == -1 && pos == NULL) || (track != -1 && pos != NULL) )
60  return -1;
61
62 if ( track != -1 ) {
63  pos = my_pos;
64  snprintf(pos, 32, "%i", track);
65 }
66
67 if ( (pid = fork()) == -1 ) {
68  return -1;
69 }
70
71 if ( pid )
72  return pid;
73
74 fh[0] = dup(cdrom);
75 fh[1] = dup(data);
76
77 if ( fh[0] == -1 || fh[1] == -1 ) {
78  ROAR_CDROM_ERROR_NORETURN("Can not dup(): %s", strerror(errno));
79 }
80
81 close(ROAR_STDIN);
82 close(ROAR_STDOUT);
83
84 // TODO: should I close some other handles?
85
86 if ( dup2(fh[0], ROAR_STDIN) == -1 || dup2(fh[1], ROAR_STDOUT) == -1 ) {
87  ROAR_CDROM_ERROR_NORETURN("Can not dup2(): %s", strerror(errno));
88 }
89
90 // new close our backups:
91 close(fh[0]);
92 close(fh[1]);
93
94 execl(ROAR_HAVE_BIN_CDPARANOIA, "cdparanoia", "--force-cdrom-device", "/dev/stdin", "-q",
95                ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT, pos, "-", NULL);
96
97 ROAR_CDROM_ERROR_NORETURN("We are still alive after exec()!, very bad!, error was: %s", strerror(errno));
98 return -1;
99#else
100#ifndef ROAR_HAVE_BIN_CDPARANOIA
101 ROAR_ERR("roar_cdrom_run_cdparanoia(*): ROAR_HAVE_BIN_CDPARANOIA not defined!");
102#endif
103#ifndef ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT
104 ROAR_ERR("roar_cdrom_run_cdparanoia(*): ROAR_CDROM_CDPARANOIA_OUTPUTFORMAT not defined!");
105#endif
106 ROAR_ERR("roar_cdrom_run_cdparanoia(cdrom=%i, data=%i, track=%i, pos='%s') = -1 // no cdparanoia support compiled in",
107             cdrom, data, track, pos);
108 return -1;
109#endif
110}
111
112int roar_cdrom_open (struct roar_connection * con, struct roar_cdrom * cdrom, char * device) {
113#ifdef ROAR_HAVE_CDROM
114 int flags;
115
116 if ( cdrom == NULL )
117  return -1;
118
119 memset((void*)cdrom, 0, sizeof(struct roar_cdrom));
120
121 if ( device == NULL )
122  device = roar_cdromdevice();
123
124 if ( device == NULL )
125  return -1;
126
127 strncpy(cdrom->device, device, ROAR_CDROM_MAX_DEVLEN);
128
129 cdrom->con        = con; // we do not care here if it is set or not as we can operate in local only mode
130
131 cdrom->stream     = -1;
132 cdrom->play_local =  1;
133 cdrom->player     = -1;
134
135 if ( (cdrom->fh = open(cdrom->device, O_RDONLY, 0644)) == -1 )
136  return -1;
137
138#ifndef ROAR_TARGET_WIN32
139 if ( (flags = fcntl(cdrom->fh, F_GETFL, 0)) == -1 ) {
140  close(cdrom->fh);
141  cdrom->fh  = -1;
142  return -1;
143 }
144
145 flags |= FD_CLOEXEC;
146
147 if ( fcntl(cdrom->fh, F_SETFL, flags) == -1 ) {
148  close(cdrom->fh);
149  cdrom->fh = -1;
150  return -1;
151 }
152#endif
153
154 return 0;
155#else
156 return -1;
157#endif
158}
159
160int roar_cdrom_close(struct roar_cdrom * cdrom) {
161#ifdef ROAR_HAVE_CDROM
162 if ( cdrom == NULL )
163  return -1;
164
165 roar_cdrom_stop(cdrom); // stop on close
166
167 if ( cdrom->fh != -1 )
168  close(cdrom->fh);
169
170 memset((void*)cdrom, 0, sizeof(struct roar_cdrom));
171
172 return 0;
173#else
174 return -1;
175#endif
176}
177
178int roar_cdrom_stop (struct roar_cdrom * cdrom) {
179#ifdef ROAR_HAVE_CDROM
180 int ret;
181
182 if ( cdrom == NULL )
183  return -1;
184
185 if ( cdrom->con == NULL )
186  return -1;
187
188 if ( cdrom->stream == -1 )
189  return -1;
190
191 if ( (ret = roar_kick(cdrom->con, ROAR_OT_STREAM, cdrom->stream)) == -1 ) {
192  return -1;
193 }
194
195#ifndef ROAR_TARGET_WIN32
196 if ( cdrom->player != -1 )
197  kill(cdrom->player, SIGINT);
198#else
199 if ( cdrom->player != -1 ) {
200  ROAR_ERR("roar_cdrom_stop(*): Can not kill player with pid %i, not supported on win32", cdrom->player);
201 }
202#endif
203
204 cdrom->player = -1;
205 cdrom->stream = -1;
206
207 return ret;
208#else
209 return -1;
210#endif
211}
212
213int roar_cdrom_play (struct roar_cdrom * cdrom, int track) {
214#ifdef ROAR_HAVE_CDROM
215 int stream_fh;
216 struct roar_stream stream[1];
217
218 if ( cdrom == NULL )
219  return -1;
220
221 if ( cdrom->con == NULL )
222  return -1;
223
224 if ( cdrom->stream != -1 ) {
225  if ( roar_cdrom_stop(cdrom) == -1 )
226   return -1;
227 }
228
229 if ( cdrom->play_local ) {
230
231  if ( (stream_fh = roar_simple_new_stream_obj(cdrom->con, stream, ROAR_CDROM_STREAMINFO, ROAR_DIR_PLAY)) == -1 ) {
232   return -1;
233  }
234
235  if ( (cdrom->player = roar_cdrom_run_cdparanoia(cdrom->fh, stream_fh, track, NULL)) != -1 ) {
236   cdrom->stream = stream->id;
237   return 0;
238  }
239
240  close(stream_fh);
241
242  return -1;
243 } else {
244  // no support for remote playback yet
245  return -1;
246 }
247#else
248 return -1;
249#endif
250}
251
252//ll
Note: See TracBrowser for help on using the repository browser.