1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
3 * Sample CDAUDIO Wine Driver for Linux
5 * Copyright 1994 Martin Ayotte
6 * Copyright 1999 Eric Pouech
12 #include <sys/ioctl.h>
15 #include "debugtools.h"
17 DEFAULT_DEBUG_CHANNEL(cdaudio)
19 #define MAX_CDAUDIO_TRACKS 256
21 /**************************************************************************
22 * CDAUDIO_Open [internal]
25 * or -1 (figure it out)
27 int CDAUDIO_Open(WINE_CDAUDIO* wcda, int drive)
35 for (i=0; i < MAX_DOS_DRIVES; i++)
36 if (DRIVE_GetType(i) == TYPE_CDROM)
48 WARN("No CD-ROM #%d found !\n", drive);
51 if ((dev = DRIVE_GetDevice(drive)) == NULL)
53 WARN("No device entry for CD-ROM #%d (drive %c:) found !\n",
58 wcda->unixdev = open(dev, O_RDONLY | O_NONBLOCK, 0);
59 if (wcda->unixdev == -1) {
60 WARN("can't open '%s'!. errno=%d\n", dev, errno);
63 wcda->cdaMode = WINE_CDA_OPEN; /* to force reading tracks info */
66 wcda->dwFirstFrame = 0;
67 wcda->dwLastFrame = 0;
68 wcda->lpdwTrackLen = NULL;
69 wcda->lpdwTrackPos = NULL;
70 wcda->lpbTrackFlags = NULL;
74 /**************************************************************************
75 * CDAUDIO_GetMediaType [internal]
77 int CDAUDIO_GetMediaType(WINE_CDAUDIO* wcda)
80 return ioctl(wcda->unixdev, CDROM_DISC_STATUS);
86 /**************************************************************************
87 * CDAUDIO_Close [internal]
89 int CDAUDIO_Close(WINE_CDAUDIO* wcda)
91 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
92 if (wcda->lpdwTrackLen != NULL) free(wcda->lpdwTrackLen);
93 if (wcda->lpdwTrackPos != NULL) free(wcda->lpdwTrackPos);
94 if (wcda->lpbTrackFlags != NULL) free(wcda->lpbTrackFlags);
102 /**************************************************************************
103 * CDAUDIO_GetNumberOfTracks [internal]
105 UINT16 CDAUDIO_GetNumberOfTracks(WINE_CDAUDIO* wcda)
107 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
109 struct cdrom_tochdr hdr;
111 struct ioc_toc_header hdr;
114 if (wcda->nTracks == 0) {
116 if (ioctl(wcda->unixdev, CDROMREADTOCHDR, &hdr))
118 if (ioctl(wcda->unixdev, CDIOREADTOCHEADER, &hdr))
121 WARN("(%p) -- Error occured (%d)!\n", wcda, errno);
125 wcda->nFirstTrack = hdr.cdth_trk0;
126 wcda->nLastTrack = hdr.cdth_trk1;
128 wcda->nFirstTrack = hdr.starting_track;
129 wcda->nLastTrack = hdr.ending_track;
131 wcda->nTracks = wcda->nLastTrack - wcda->nFirstTrack + 1;
133 return wcda->nTracks;
139 /**************************************************************************
140 * CDAUDIO_GetTracksInfo [internal]
142 BOOL CDAUDIO_GetTracksInfo(WINE_CDAUDIO* wcda)
144 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
146 int start, last_start = 0;
147 int total_length = 0;
149 struct cdrom_tocentry entry;
151 struct ioc_read_toc_entry entry;
152 struct cd_toc_entry toc_buffer;
155 if (wcda->nTracks == 0) {
156 if (CDAUDIO_GetNumberOfTracks(wcda) == (WORD)-1) return FALSE;
158 TRACE("nTracks=%u\n", wcda->nTracks);
160 if (wcda->lpdwTrackLen != NULL)
161 free(wcda->lpdwTrackLen);
162 wcda->lpdwTrackLen = (LPDWORD)malloc((wcda->nTracks + 1) * sizeof(DWORD));
163 if (wcda->lpdwTrackPos != NULL)
164 free(wcda->lpdwTrackPos);
165 wcda->lpdwTrackPos = (LPDWORD)malloc((wcda->nTracks + 1) * sizeof(DWORD));
166 if (wcda->lpbTrackFlags != NULL)
167 free(wcda->lpbTrackFlags);
168 wcda->lpbTrackFlags = (LPBYTE)malloc((wcda->nTracks + 1) * sizeof(BYTE));
169 if (wcda->lpdwTrackLen == NULL || wcda->lpdwTrackPos == NULL ||
170 wcda->lpbTrackFlags == NULL) {
171 WARN("error allocating track table !\n");
174 memset(wcda->lpdwTrackLen, 0, (wcda->nTracks + 1) * sizeof(DWORD));
175 memset(wcda->lpdwTrackPos, 0, (wcda->nTracks + 1) * sizeof(DWORD));
176 memset(wcda->lpbTrackFlags, 0, (wcda->nTracks + 1) * sizeof(BYTE));
177 for (i = 0; i <= wcda->nTracks; i++) {
178 if (i == wcda->nTracks)
180 entry.cdte_track = CDROM_LEADOUT;
183 entry.starting_track = LEADOUT; /* XXX */
187 entry.cdte_track = i + 1;
189 entry.starting_track = i + 1;
192 entry.cdte_format = CDROM_MSF;
194 bzero((char *)&toc_buffer, sizeof(toc_buffer));
195 entry.address_format = CD_MSF_FORMAT;
196 entry.data_len = sizeof(toc_buffer);
197 entry.data = &toc_buffer;
200 if (ioctl(wcda->unixdev, CDROMREADTOCENTRY, &entry))
202 if (ioctl(wcda->unixdev, CDIOREADTOCENTRYS, &entry))
205 WARN("error read entry (%d)\n", errno);
206 /* update status according to new status */
207 CDAUDIO_GetCDStatus(wcda);
212 start = CDFRAMES_PERSEC * (SECONDS_PERMIN *
213 entry.cdte_addr.msf.minute + entry.cdte_addr.msf.second) +
214 entry.cdte_addr.msf.frame;
216 start = CDFRAMES_PERSEC * (SECONDS_PERMIN *
217 toc_buffer.addr.msf.minute + toc_buffer.addr.msf.second) +
218 toc_buffer.addr.msf.frame;
222 wcda->dwFirstFrame = start;
223 TRACE("dwFirstOffset=%u\n", start);
225 length = start - last_start;
227 start = last_start - length;
228 total_length += length;
229 wcda->lpdwTrackLen[i - 1] = length;
230 wcda->lpdwTrackPos[i - 1] = start;
231 TRACE("track #%u start=%u len=%u\n", i, start, length);
234 wcda->lpbTrackFlags[i] =
235 (entry.cdte_adr << 4) | (entry.cdte_ctrl & 0x0f);
237 wcda->lpbTrackFlags[i] =
238 (toc_buffer.addr_type << 4) | (toc_buffer.control & 0x0f);
240 TRACE("track #%u flags=%02x\n", i + 1, wcda->lpbTrackFlags[i]);
242 wcda->dwLastFrame = last_start;
243 TRACE("total_len=%u\n", total_length);
250 /**************************************************************************
251 * CDAUDIO_GetCDStatus [internal]
253 BOOL CDAUDIO_GetCDStatus(WINE_CDAUDIO* wcda)
255 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
256 int oldmode = wcda->cdaMode;
258 wcda->sc.cdsc_format = CDROM_MSF;
260 struct ioc_read_subchannel read_sc;
262 read_sc.address_format = CD_MSF_FORMAT;
263 read_sc.data_format = CD_CURRENT_POSITION;
265 read_sc.data_len = sizeof(wcda->sc);
266 read_sc.data = (struct cd_sub_channel_info *)&wcda->sc;
269 if (ioctl(wcda->unixdev, CDROMSUBCHNL, &wcda->sc))
271 if (ioctl(wcda->unixdev, CDIOCREADSUBCHANNEL, &read_sc))
274 TRACE("opened or no_media (%d)!\n", errno);
275 wcda->cdaMode = WINE_CDA_OPEN; /* was NOT_READY */
280 wcda->sc.cdsc_audiostatus
282 wcda->sc.header.audio_status
286 case CDROM_AUDIO_INVALID:
288 case CD_AS_AUDIO_INVALID:
290 WARN("device doesn't support status.\n");
291 wcda->cdaMode = WINE_CDA_DONTKNOW;
294 case CDROM_AUDIO_NO_STATUS:
296 case CD_AS_NO_STATUS:
298 wcda->cdaMode = WINE_CDA_STOP;
299 TRACE("WINE_CDA_STOP !\n");
302 case CDROM_AUDIO_PLAY:
304 case CD_AS_PLAY_IN_PROGRESS:
306 wcda->cdaMode = WINE_CDA_PLAY;
309 case CDROM_AUDIO_PAUSED:
311 case CD_AS_PLAY_PAUSED:
313 wcda->cdaMode = WINE_CDA_PAUSE;
314 TRACE("WINE_CDA_PAUSE !\n");
318 TRACE("status=%02X !\n",
319 wcda->sc.cdsc_audiostatus);
321 TRACE("status=%02X !\n",
322 wcda->sc.header.audio_status);
326 wcda->nCurTrack = wcda->sc.cdsc_trk;
328 CDFRAMES_PERMIN * wcda->sc.cdsc_absaddr.msf.minute +
329 CDFRAMES_PERSEC * wcda->sc.cdsc_absaddr.msf.second +
330 wcda->sc.cdsc_absaddr.msf.frame;
332 wcda->nCurTrack = wcda->sc.what.position.track_number;
334 CDFRAMES_PERMIN * wcda->sc.what.position.absaddr.msf.minute +
335 CDFRAMES_PERSEC * wcda->sc.what.position.absaddr.msf.second +
336 wcda->sc.what.position.absaddr.msf.frame;
339 TRACE("%02u-%02u:%02u:%02u \n",
341 wcda->sc.cdsc_absaddr.msf.minute,
342 wcda->sc.cdsc_absaddr.msf.second,
343 wcda->sc.cdsc_absaddr.msf.frame);
345 TRACE("%02u-%02u:%02u:%02u \n",
346 wcda->sc.what.position.track_number,
347 wcda->sc.what.position.absaddr.msf.minute,
348 wcda->sc.what.position.absaddr.msf.second,
349 wcda->sc.what.position.absaddr.msf.frame);
352 if (oldmode != wcda->cdaMode && oldmode == WINE_CDA_OPEN) {
353 if (!CDAUDIO_GetTracksInfo(wcda)) {
354 WARN("error updating TracksInfo !\n");
364 /**************************************************************************
365 * CDAUDIO_Play [internal]
367 int CDAUDIO_Play(WINE_CDAUDIO* wcda, DWORD start, DWORD end)
369 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
371 struct cdrom_msf msf;
373 struct ioc_play_msf msf;
377 msf.cdmsf_min0 = start / CDFRAMES_PERMIN;
378 msf.cdmsf_sec0 = (start % CDFRAMES_PERMIN) / CDFRAMES_PERSEC;
379 msf.cdmsf_frame0 = start % CDFRAMES_PERSEC;
380 msf.cdmsf_min1 = end / CDFRAMES_PERMIN;
381 msf.cdmsf_sec1 = (end % CDFRAMES_PERMIN) / CDFRAMES_PERSEC;
382 msf.cdmsf_frame1 = end % CDFRAMES_PERSEC;
384 msf.start_m = start / CDFRAMES_PERMIN;
385 msf.start_s = (start % CDFRAMES_PERMIN) / CDFRAMES_PERSEC;
386 msf.start_f = start % CDFRAMES_PERSEC;
387 msf.end_m = end / CDFRAMES_PERMIN;
388 msf.end_s = (end % CDFRAMES_PERMIN) / CDFRAMES_PERSEC;
389 msf.end_f = end % CDFRAMES_PERSEC;
392 if (ioctl(wcda->unixdev, CDROMSTART))
394 if (ioctl(wcda->unixdev, CDIOCSTART, NULL))
397 WARN("motor doesn't start !\n");
401 if (ioctl(wcda->unixdev, CDROMPLAYMSF, &msf))
403 if (ioctl(wcda->unixdev, CDIOCPLAYMSF, &msf))
406 WARN("device doesn't play !\n");
410 TRACE("msf = %d:%d:%d %d:%d:%d\n",
411 msf.cdmsf_min0, msf.cdmsf_sec0, msf.cdmsf_frame0,
412 msf.cdmsf_min1, msf.cdmsf_sec1, msf.cdmsf_frame1);
414 TRACE("msf = %d:%d:%d %d:%d:%d\n",
415 msf.start_m, msf.start_s, msf.start_f,
416 msf.end_m, msf.end_s, msf.end_f);
424 /**************************************************************************
425 * CDAUDIO_Stop [internal]
427 int CDAUDIO_Stop(WINE_CDAUDIO* wcda)
429 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
432 ret = ioctl(wcda->unixdev, CDROMSTOP);
434 ret = ioctl(wcda->unixdev, CDIOCSTOP, NULL);
442 /**************************************************************************
443 * CDAUDIO_Pause [internal]
445 int CDAUDIO_Pause(WINE_CDAUDIO* wcda, int pauseOn)
447 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
450 ret = ioctl(wcda->unixdev, pauseOn ? CDROMPAUSE : CDROMRESUME);
452 ret = ioctl(wcda->unixdev, pauseOn ? CDIOCPAUSE : CDIOCRESUME, NULL);
460 /**************************************************************************
461 * CDAUDIO_Seek [internal]
463 int CDAUDIO_Seek(WINE_CDAUDIO* wcda, DWORD at)
465 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
468 struct cdrom_msf0 msf;
469 msf.minute = at / CDFRAMES_PERMIN;
470 msf.second = (at % CDFRAMES_PERMIN) / CDFRAMES_PERSEC;
471 msf.frame = at % CDFRAMES_PERSEC;
473 ret = ioctl(wcda->unixdev, CDROMSEEK, &msf);
475 /* FIXME: the current end for play is lost
476 * use end of CD ROM instead
478 FIXME("Could a BSD expert implement the seek function ?\n");
479 CDAUDIO_Play(wcda, at, wcda->lpdwTrackPos[wcda->nTracks] + wcda->lpdwTrackLen[wcda->nTracks]);
488 /**************************************************************************
489 * CDAUDIO_SetDoor [internal]
491 int CDAUDIO_SetDoor(WINE_CDAUDIO* wcda, int open)
493 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
497 ret = ioctl(wcda->unixdev, CDROMEJECT);
499 ret = ioctl(wcda->unixdev, CDROMEJECT, 1);
502 ret = (ioctl(wcda->unixdev, CDIOCALLOW, NULL)) ||
503 (ioctl(wcda->unixdev, open ? CDIOCEJECT : CDIOCCLOSE, NULL)) ||
504 (ioctl(wcda->unixdev, CDIOCPREVENT, NULL));
513 /**************************************************************************
514 * CDAUDIO_Reset [internal]
516 int CDAUDIO_Reset(WINE_CDAUDIO* wcda)
518 #if defined(linux) || defined(__FreeBSD__) || defined(__NetBSD__)
521 ret = ioctl(wcda->unixdev, CDROMRESET);
523 ret = ioctl(wcda->unixdev, CDIOCRESET, NULL);