src/cdrom/osf/SDL_syscdrom.c
author Sam Lantinga <slouken@libsdl.org>
Tue, 07 Feb 2006 06:59:48 +0000
changeset 1336 3692456e7b0f
parent 1019 e3b3130f3af8
child 1338 604d73db6802
permissions -rw-r--r--
Use SDL_ prefixed versions of C library functions.
FIXME:
Change #include <stdlib.h> to #include "SDL_stdlib.h"
Change #include <string.h> to #include "SDL_string.h"
Make sure nothing else broke because of this...
slouken@654
     1
/*
slouken@654
     2
    Tru64 audio module for SDL (Simple DirectMedia Layer)
slouken@654
     3
    Copyright (C) 2003
slouken@654
     4
slouken@654
     5
    This library is free software; you can redistribute it and/or
slouken@654
     6
    modify it under the terms of the GNU Library General Public
slouken@654
     7
    License as published by the Free Software Foundation; either
slouken@654
     8
    version 2 of the License, or (at your option) any later version.
slouken@654
     9
slouken@654
    10
    This library is distributed in the hope that it will be useful,
slouken@654
    11
    but WITHOUT ANY WARRANTY; without even the implied warranty of
slouken@654
    12
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
slouken@654
    13
    Library General Public License for more details.
slouken@654
    14
slouken@654
    15
    You should have received a copy of the GNU Library General Public
slouken@654
    16
    License along with this library; if not, write to the Free
slouken@654
    17
    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
slouken@654
    18
slouken@654
    19
slouken@654
    20
*/
slouken@654
    21
slouken@654
    22
slouken@654
    23
/* Functions for system-level CD-ROM audio control */
slouken@654
    24
slouken@1019
    25
/* #define DEBUG_CDROM 1 */
slouken@654
    26
slouken@654
    27
#include <sys/types.h>
slouken@1019
    28
#include <dirent.h>
slouken@654
    29
#include <sys/stat.h>
slouken@654
    30
#include <fcntl.h>
slouken@654
    31
#include <io/cam/cdrom.h>
slouken@654
    32
#include <io/cam/rzdisk.h>
slouken@654
    33
#include <io/common/devgetinfo.h>
slouken@654
    34
#include <alloca.h>
slouken@654
    35
#include <stdlib.h>
slouken@654
    36
#include <stdio.h>
slouken@654
    37
#include <string.h>
slouken@654
    38
#include <errno.h>
slouken@654
    39
slouken@654
    40
#include "SDL_error.h"
slouken@654
    41
#include "SDL_cdrom.h"
slouken@654
    42
#include "SDL_syscdrom.h"
slouken@654
    43
slouken@654
    44
/* The maximum number of CD-ROM drives we'll detect */
slouken@654
    45
#define MAX_DRIVES 16
slouken@654
    46
slouken@654
    47
/* A list of available CD-ROM drives */
slouken@654
    48
static char *SDL_cdlist[MAX_DRIVES];
slouken@654
    49
static dev_t SDL_cdmode[MAX_DRIVES];
slouken@654
    50
slouken@654
    51
/* The system-dependent CD control functions */
slouken@654
    52
static const char *SDL_SYS_CDName(int drive);
slouken@654
    53
static int         SDL_SYS_CDOpen(int drive);
slouken@654
    54
static int         SDL_SYS_CDGetTOC(SDL_CD *cdrom);
slouken@654
    55
static CDstatus    SDL_SYS_CDStatus(SDL_CD *cdrom, int *position);
slouken@654
    56
static int         SDL_SYS_CDPlay(SDL_CD *cdrom, int start, int length);
slouken@654
    57
static int         SDL_SYS_CDPause(SDL_CD *cdrom);
slouken@654
    58
static int         SDL_SYS_CDResume(SDL_CD *cdrom);
slouken@654
    59
static int         SDL_SYS_CDStop(SDL_CD *cdrom);
slouken@654
    60
static int         SDL_SYS_CDEject(SDL_CD *cdrom);
slouken@654
    61
static void        SDL_SYS_CDClose(SDL_CD *cdrom);
slouken@654
    62
slouken@654
    63
/* Check a drive to see if it is a CD-ROM */
slouken@654
    64
/* Caution!! Not tested. */ 
slouken@654
    65
static int CheckDrive(char *drive, struct stat *stbuf)
slouken@654
    66
{
slouken@873
    67
    int cdfd, is_cd = 0;
slouken@873
    68
    struct mode_sel_sns_params msp;
slouken@873
    69
    struct inquiry_info inq;
slouken@873
    70
slouken@873
    71
#ifdef DEBUG_CDROM
slouken@873
    72
    char *devtype[] = {"Disk", "Tape", "Printer", "Processor", "WORM",
slouken@873
    73
	"CD-ROM", "Scanner", "Optical", "Changer", "Comm", "Unknown"};
slouken@873
    74
#endif
slouken@873
    75
slouken@873
    76
    bzero(&msp, sizeof(msp));
slouken@873
    77
    bzero(&inq, sizeof(inq));
slouken@654
    78
slouken@654
    79
    /* If it doesn't exist, return -1 */
slouken@654
    80
    if ( stat(drive, stbuf) < 0 ) {
slouken@654
    81
	return(-1);
slouken@654
    82
    }
slouken@654
    83
slouken@873
    84
    if ( (cdfd = open(drive, (O_RDWR|O_NDELAY), 0)) >= 0 ) {
slouken@873
    85
	msp.msp_addr   =   (caddr_t) &inq;
slouken@873
    86
	msp.msp_pgcode =                0;
slouken@873
    87
	msp.msp_pgctrl =                0;
slouken@873
    88
	msp.msp_length =      sizeof(inq);
slouken@873
    89
	msp.msp_setps  =                0;
slouken@654
    90
slouken@873
    91
	if ( ioctl(cdfd, SCSI_GET_INQUIRY_DATA, &msp) )
slouken@873
    92
	    return (0);
slouken@873
    93
slouken@873
    94
#ifdef DEBUG_CDROM
slouken@873
    95
	fprintf(stderr, "Device Type: %s\n", devtype[inq.perfdt]);
slouken@873
    96
	fprintf(stderr, "Vendor: %.8s\n", inq.vndrid);
slouken@873
    97
	fprintf(stderr, "Product: %.8s\n", inq.prodid);
slouken@873
    98
	fprintf(stderr, "Revision: %.8s\n", inq.revlvl);
slouken@873
    99
#endif
slouken@873
   100
	if ( inq.perfdt == DTYPE_RODIRECT )
slouken@873
   101
	    is_cd = 1;
slouken@654
   102
    }
slouken@654
   103
slouken@654
   104
    return(is_cd);
slouken@654
   105
}
slouken@654
   106
slouken@654
   107
/* Add a CD-ROM drive to our list of valid drives */
slouken@654
   108
static void AddDrive(char *drive, struct stat *stbuf)
slouken@654
   109
{
slouken@654
   110
    int i;
slouken@654
   111
slouken@654
   112
    if ( SDL_numcds < MAX_DRIVES ) {
slouken@654
   113
	/* Check to make sure it's not already in our list.
slouken@654
   114
	 * This can happen when we see a drive via symbolic link.
slouken@654
   115
	 *
slouken@873
   116
	 */
slouken@654
   117
	for ( i=0; i<SDL_numcds; ++i ) {
slouken@654
   118
	    if ( stbuf->st_rdev == SDL_cdmode[i] ) {
slouken@654
   119
#ifdef DEBUG_CDROM
slouken@654
   120
  fprintf(stderr, "Duplicate drive detected: %s == %s\n", drive, SDL_cdlist[i]);
slouken@654
   121
#endif
slouken@654
   122
	    return;
slouken@654
   123
	    }
slouken@654
   124
	}
slouken@654
   125
slouken@654
   126
	/* Add this drive to our list */
slouken@654
   127
	i = SDL_numcds;
slouken@1336
   128
	SDL_cdlist[i] = (char *)SDL_malloc(SDL_strlen(drive)+1);
slouken@654
   129
	if ( SDL_cdlist[i] == NULL ) {
slouken@654
   130
	    SDL_OutOfMemory();
slouken@654
   131
	    return;
slouken@654
   132
	}
slouken@654
   133
slouken@1336
   134
	SDL_strcpy(SDL_cdlist[i], drive);
slouken@654
   135
	SDL_cdmode[i] = stbuf->st_rdev;
slouken@654
   136
	++SDL_numcds;
slouken@654
   137
#ifdef DEBUG_CDROM
slouken@654
   138
  fprintf(stderr, "Added CD-ROM drive: %s\n", drive);
slouken@654
   139
#endif
slouken@654
   140
    }
slouken@654
   141
}
slouken@654
   142
slouken@654
   143
int  SDL_SYS_CDInit(void)
slouken@654
   144
{
slouken@1019
   145
    /* checklist:
slouken@1019
   146
     *
slouken@1019
   147
     * Tru64 5.X (/dev/rdisk/cdrom?c)
slouken@1019
   148
     * dir: /dev/rdisk, name: cdrom
slouken@1019
   149
     *
slouken@1019
   150
     * Digital UNIX 4.0X (/dev/rrz?c)
slouken@1019
   151
     * dir: /dev, name: rrz
slouken@654
   152
     *
slouken@654
   153
     */
slouken@1019
   154
    struct {
slouken@1019
   155
	char *dir;
slouken@1019
   156
	char *name;
slouken@1019
   157
    } checklist[] = {
slouken@1019
   158
	{"/dev/rdisk", "cdrom"},
slouken@1019
   159
	{"/dev", "rrz"},
slouken@1019
   160
	{NULL, NULL}};
slouken@654
   161
    char drive[32];
slouken@654
   162
    char *SDLcdrom;
slouken@654
   163
    int i, j, exists;
slouken@654
   164
    struct stat stbuf;
slouken@654
   165
slouken@654
   166
    /* Fill in our driver capabilities */
slouken@654
   167
    SDL_CDcaps.Name   = SDL_SYS_CDName;
slouken@654
   168
    SDL_CDcaps.Open   = SDL_SYS_CDOpen;
slouken@654
   169
    SDL_CDcaps.GetTOC = SDL_SYS_CDGetTOC;
slouken@654
   170
    SDL_CDcaps.Status = SDL_SYS_CDStatus;
slouken@654
   171
    SDL_CDcaps.Play   = SDL_SYS_CDPlay;
slouken@654
   172
    SDL_CDcaps.Pause  = SDL_SYS_CDPause;
slouken@654
   173
    SDL_CDcaps.Resume = SDL_SYS_CDResume;
slouken@654
   174
    SDL_CDcaps.Stop   = SDL_SYS_CDStop;
slouken@654
   175
    SDL_CDcaps.Eject  = SDL_SYS_CDEject;
slouken@654
   176
    SDL_CDcaps.Close  = SDL_SYS_CDClose;
slouken@654
   177
slouken@654
   178
slouken@654
   179
    /* Look in the environment for our CD-ROM drive list */
slouken@1336
   180
    SDLcdrom = SDL_getenv("SDL_CDROM");	/* ':' separated list of devices */
slouken@654
   181
    if ( SDLcdrom != NULL ) {
slouken@654
   182
	char *cdpath, *delim;
slouken@1336
   183
	cdpath = SDL_malloc(SDL_strlen(SDLcdrom)+1);
slouken@654
   184
	if ( cdpath != NULL ) {
slouken@1336
   185
	    SDL_strcpy(cdpath, SDLcdrom);
slouken@654
   186
	    SDLcdrom = cdpath;
slouken@654
   187
	    do {
slouken@1336
   188
		delim = SDL_strchr(SDLcdrom, ':');
slouken@654
   189
		if ( delim ) {
slouken@654
   190
		    *delim++ = '\0';
slouken@654
   191
		}
slouken@654
   192
		if ( CheckDrive(SDLcdrom, &stbuf) > 0 ) {
slouken@654
   193
		    AddDrive(SDLcdrom, &stbuf);
slouken@654
   194
		}
slouken@654
   195
		if ( delim ) {
slouken@654
   196
		    SDLcdrom = delim;
slouken@654
   197
		} else {
slouken@654
   198
		    SDLcdrom = NULL;
slouken@654
   199
		}
slouken@654
   200
	    } while ( SDLcdrom );
slouken@1336
   201
	    SDL_free(cdpath);
slouken@654
   202
	}
slouken@654
   203
slouken@654
   204
	/* If we found our drives, there's nothing left to do */
slouken@654
   205
	if ( SDL_numcds > 0 ) {
slouken@654
   206
	    return(0);
slouken@654
   207
	}
slouken@654
   208
    }
slouken@654
   209
    /* Scan the system for CD-ROM drives */
slouken@1019
   210
    for ( i = 0; checklist[i].dir; ++i) {
slouken@1019
   211
	DIR *devdir;
slouken@1019
   212
	struct dirent *devent;
slouken@1019
   213
	int name_len;
slouken@1019
   214
slouken@1019
   215
	devdir = opendir(checklist[i].dir);
slouken@1019
   216
	if (devdir) {
slouken@1336
   217
	    name_len = SDL_strlen(checklist[i].name);
slouken@1019
   218
	    while (devent = readdir(devdir))
slouken@1336
   219
		if (SDL_memcmp(checklist[i].name, devent->d_name, name_len) == 0)
slouken@1019
   220
		    if (devent->d_name[devent->d_namlen-1] == 'c') {
slouken@1019
   221
			sprintf(drive, "%s/%s", checklist[i].dir, devent->d_name);
slouken@1019
   222
#ifdef DEBUG_CDROM
slouken@1019
   223
			fprintf(stderr, "Try to add drive: %s\n", drive);
slouken@1019
   224
#endif
slouken@1019
   225
			if ( CheckDrive(drive, &stbuf) > 0 )
slouken@1019
   226
			    AddDrive(drive, &stbuf);
slouken@1019
   227
		    }
slouken@1019
   228
	    closedir(devdir);
slouken@654
   229
	} else {
slouken@1019
   230
#ifdef DEBUG_CDROM
slouken@1019
   231
	    fprintf(stderr, "cannot open dir: %s\n", checklist[i].dir);
slouken@1019
   232
#endif
slouken@654
   233
	}
slouken@654
   234
    }
slouken@1019
   235
slouken@654
   236
/*
slouken@1336
   237
    SDLcdrom=SDL_malloc(sizeof(char) * 32);
slouken@1336
   238
    SDL_strcpy(SDLcdrom,"/dev/rdisk/cdrom0c");
slouken@654
   239
    SDL_cdlist[0] = SDLcdrom;
slouken@654
   240
    stat(SDLcdrom, &stbuf);
slouken@654
   241
    SDL_cdmode[0] = stbuf.st_rdev;
slouken@654
   242
    SDL_numcds = 1;
slouken@654
   243
 */
slouken@654
   244
    return (0);
slouken@654
   245
}
slouken@654
   246
slouken@654
   247
static const char *SDL_SYS_CDName(int drive)
slouken@654
   248
{
slouken@654
   249
    return(SDL_cdlist[drive]);
slouken@654
   250
}
slouken@654
   251
slouken@654
   252
static int SDL_SYS_CDOpen(int drive)
slouken@654
   253
{
slouken@654
   254
    /* O_RDWR: To use ioctl(fd, SCSI_STOP_UNIT) */
slouken@654
   255
    return(open(SDL_cdlist[drive], (O_RDWR|O_NDELAY), 0));
slouken@654
   256
}
slouken@654
   257
slouken@654
   258
static int SDL_SYS_CDGetTOC(SDL_CD *cdrom)
slouken@654
   259
{
slouken@654
   260
    struct cd_toc                  toc;
slouken@654
   261
    struct cd_toc_header           hdr;
slouken@654
   262
    struct cd_toc_entry          *cdte;
slouken@654
   263
    int i;
slouken@654
   264
    int okay = 0;
slouken@654
   265
    if ( ioctl(cdrom->id, CDROM_TOC_HEADER, &hdr) ) {
slouken@654
   266
	fprintf(stderr,"ioctl error CDROM_TOC_HEADER\n");
slouken@654
   267
	return -1;
slouken@654
   268
    }
slouken@654
   269
    cdrom->numtracks = hdr.th_ending_track - hdr.th_starting_track + 1;
slouken@654
   270
    if ( cdrom->numtracks > SDL_MAX_TRACKS ) {
slouken@654
   271
	cdrom->numtracks = SDL_MAX_TRACKS;
slouken@654
   272
    }
slouken@654
   273
#ifdef DEBUG_CDROM
slouken@654
   274
  fprintf(stderr,"hdr.th_data_len1 = %d\n", hdr.th_data_len1);
slouken@654
   275
  fprintf(stderr,"hdr.th_data_len0 = %d\n", hdr.th_data_len0);
slouken@654
   276
  fprintf(stderr,"hdr.th_starting_track = %d\n", hdr.th_starting_track);
slouken@654
   277
  fprintf(stderr,"hdr.th_ending_track = %d\n", hdr.th_ending_track);
slouken@654
   278
  fprintf(stderr,"cdrom->numtracks = %d\n", cdrom->numtracks);
slouken@654
   279
#endif
slouken@654
   280
    toc.toc_address_format = CDROM_LBA_FORMAT;
slouken@654
   281
    toc.toc_starting_track = 0;
slouken@654
   282
    toc.toc_alloc_length = (hdr.th_data_len1 << 8) +
slouken@654
   283
			    hdr.th_data_len0 + sizeof(hdr);
slouken@654
   284
    if ( (toc.toc_buffer = alloca(toc.toc_alloc_length)) == NULL) {
slouken@654
   285
	fprintf(stderr,"cannot allocate toc.toc_buffer\n");
slouken@654
   286
	return -1;
slouken@654
   287
    }
slouken@654
   288
slouken@654
   289
    bzero (toc.toc_buffer, toc.toc_alloc_length);
slouken@654
   290
    if (ioctl(cdrom->id, CDROM_TOC_ENTRYS, &toc)) {
slouken@654
   291
	fprintf(stderr,"ioctl error CDROM_TOC_ENTRYS\n");
slouken@654
   292
	return -1;
slouken@654
   293
    }
slouken@654
   294
slouken@873
   295
    cdte =(struct cd_toc_entry *) ((char *) toc.toc_buffer + sizeof(hdr));
slouken@873
   296
    for (i=0; i <= cdrom->numtracks; ++i) {
slouken@873
   297
	if (i == cdrom->numtracks ) {
slouken@873
   298
	    cdrom->track[i].id = 0xAA;;
slouken@873
   299
	} else {
slouken@873
   300
	    cdrom->track[i].id = hdr.th_starting_track + i;
slouken@873
   301
	}
slouken@654
   302
slouken@873
   303
	cdrom->track[i].type =
slouken@873
   304
	    cdte[i].te_control & CDROM_DATA_TRACK;
slouken@873
   305
	cdrom->track[i].offset =
slouken@873
   306
	    cdte[i].te_absaddr.lba.addr3 << 24 |
slouken@873
   307
	    cdte[i].te_absaddr.lba.addr2 << 16 |
slouken@873
   308
	    cdte[i].te_absaddr.lba.addr1 << 8  |
slouken@873
   309
	    cdte[i].te_absaddr.lba.addr0;
slouken@873
   310
	cdrom->track[i].length = 0;
slouken@873
   311
	if ( i > 0 ) {
slouken@873
   312
	    cdrom->track[i - 1].length =
slouken@873
   313
		cdrom->track[i].offset -
slouken@873
   314
		cdrom->track[i - 1].offset;
slouken@654
   315
	}
slouken@873
   316
    }
slouken@654
   317
#ifdef DEBUG_CDROM
slouken@654
   318
  for (i = 0; i <= cdrom->numtracks; i++) {
slouken@654
   319
    fprintf(stderr,"toc_entry[%d].te_track_number = %d\n",
slouken@654
   320
	    i,cdte[i].te_track_number);
slouken@654
   321
    fprintf(stderr,"cdrom->track[%d].id = %d\n", i,cdrom->track[i].id);
slouken@654
   322
    fprintf(stderr,"cdrom->track[%d].type = %x\n", i,cdrom->track[i].type);
slouken@654
   323
    fprintf(stderr,"cdrom->track[%d].offset = %d\n", i,cdrom->track[i].offset);
slouken@654
   324
    fprintf(stderr,"cdrom->track[%d].length = %d\n", i,cdrom->track[i].length);
slouken@654
   325
  }
slouken@654
   326
#endif
slouken@654
   327
    if ( i == (cdrom->numtracks+1) ) {
slouken@654
   328
	okay = 1;
slouken@654
   329
    }
slouken@654
   330
slouken@654
   331
    return(okay ? 0 : -1);
slouken@654
   332
}
slouken@654
   333
slouken@654
   334
/* Get CD-ROM status */
slouken@654
   335
static CDstatus SDL_SYS_CDStatus(SDL_CD *cdrom, int *position)
slouken@654
   336
{
slouken@654
   337
    CDstatus                     status;
slouken@654
   338
    struct cd_sub_channel            sc;
slouken@654
   339
    struct cd_subc_channel_data     scd;
slouken@654
   340
slouken@654
   341
    sc.sch_address_format = CDROM_LBA_FORMAT;
slouken@654
   342
    sc.sch_data_format    = CDROM_CURRENT_POSITION;
slouken@654
   343
    sc.sch_track_number   = 0;
slouken@654
   344
    sc.sch_alloc_length   = sizeof(scd);
slouken@654
   345
    sc.sch_buffer         = (caddr_t)&scd;
slouken@654
   346
    if ( ioctl(cdrom->id, CDROM_READ_SUBCHANNEL, &sc) ) {
slouken@654
   347
	status = CD_ERROR;
slouken@654
   348
	fprintf(stderr,"ioctl error CDROM_READ_SUBCHANNEL \n");
slouken@654
   349
    } else {
slouken@654
   350
	switch (scd.scd_header.sh_audio_status) {
slouken@654
   351
	    case AS_AUDIO_INVALID:
slouken@654
   352
		status = CD_STOPPED;
slouken@654
   353
		break;
slouken@654
   354
	    case AS_PLAY_IN_PROGRESS:
slouken@654
   355
		status = CD_PLAYING;
slouken@654
   356
		break;
slouken@654
   357
	    case AS_PLAY_PAUSED:
slouken@654
   358
		status = CD_PAUSED;
slouken@654
   359
		break;
slouken@654
   360
	    case AS_PLAY_COMPLETED:
slouken@654
   361
		status = CD_STOPPED;
slouken@654
   362
		break;
slouken@654
   363
	    case AS_PLAY_ERROR:
slouken@654
   364
		status = CD_ERROR;
slouken@654
   365
		break;
slouken@654
   366
	    case AS_NO_STATUS:
slouken@654
   367
		status = CD_STOPPED;
slouken@654
   368
		break;
slouken@654
   369
	    default:
slouken@654
   370
		status = CD_ERROR;
slouken@654
   371
		break;
slouken@654
   372
	}
slouken@654
   373
#ifdef DEBUG_CDROM
slouken@654
   374
  fprintf(stderr,"scd.scd_header.sh_audio_status = %x\n",
slouken@654
   375
	scd.scd_header.sh_audio_status);
slouken@654
   376
#endif
slouken@654
   377
    }
slouken@654
   378
    if (position) {
slouken@654
   379
	if (status == CD_PLAYING || (status == CD_PAUSED) ) {
slouken@654
   380
	    *position =
slouken@654
   381
		scd.scd_position_data.scp_absaddr.lba.addr3 << 24 |
slouken@654
   382
		scd.scd_position_data.scp_absaddr.lba.addr2 << 16 |
slouken@654
   383
		scd.scd_position_data.scp_absaddr.lba.addr1 << 8  |
slouken@654
   384
		scd.scd_position_data.scp_absaddr.lba.addr0;
slouken@654
   385
	} else {
slouken@654
   386
	    *position = 0;
slouken@654
   387
	}
slouken@654
   388
    }
slouken@654
   389
slouken@654
   390
    return status;
slouken@654
   391
}
slouken@654
   392
slouken@654
   393
/* Start play */
slouken@654
   394
static int SDL_SYS_CDPlay(SDL_CD *cdrom, int start, int length)
slouken@654
   395
{
slouken@873
   396
/*
slouken@873
   397
 * Play MSF
slouken@654
   398
 */
slouken@654
   399
    struct cd_play_audio_msf msf;
slouken@654
   400
    int end;
slouken@654
   401
slouken@654
   402
    bzero(&msf, sizeof(msf));
slouken@654
   403
    end = start +length;
slouken@874
   404
    FRAMES_TO_MSF(start + 150, /* LBA = 4500*M + 75*S + F - 150 */
slouken@654
   405
		  &msf.msf_starting_M_unit,
slouken@654
   406
		  &msf.msf_starting_S_unit,
slouken@654
   407
		  &msf.msf_starting_F_unit);
slouken@874
   408
    FRAMES_TO_MSF(end + 150, /* LBA = 4500*M + 75*S + F - 150 */
slouken@654
   409
		  &msf.msf_ending_M_unit,
slouken@654
   410
		  &msf.msf_ending_S_unit,
slouken@654
   411
		  &msf.msf_ending_F_unit);
slouken@654
   412
slouken@654
   413
    return(ioctl(cdrom->id, CDROM_PLAY_AUDIO_MSF, &msf));
slouken@654
   414
}
slouken@654
   415
slouken@654
   416
/* Pause play */
slouken@654
   417
static int SDL_SYS_CDPause(SDL_CD *cdrom)
slouken@654
   418
{
slouken@654
   419
    return(ioctl(cdrom->id, CDROM_PAUSE_PLAY));
slouken@654
   420
}
slouken@654
   421
slouken@654
   422
/* Resume play */
slouken@654
   423
static int SDL_SYS_CDResume(SDL_CD *cdrom)
slouken@654
   424
{
slouken@654
   425
    return(ioctl(cdrom->id, CDROM_RESUME_PLAY));
slouken@654
   426
}
slouken@654
   427
slouken@654
   428
/* Stop play */
slouken@654
   429
static int SDL_SYS_CDStop(SDL_CD *cdrom)
slouken@654
   430
{
slouken@654
   431
    return(ioctl(cdrom->id, SCSI_STOP_UNIT));
slouken@654
   432
}
slouken@654
   433
slouken@654
   434
/* Eject the CD-ROM */
slouken@654
   435
static int SDL_SYS_CDEject(SDL_CD *cdrom)
slouken@654
   436
{
slouken@654
   437
    return(ioctl(cdrom->id, CDROM_EJECT_CADDY));
slouken@654
   438
}
slouken@654
   439
slouken@654
   440
/* Close the CD-ROM handle */
slouken@654
   441
static void SDL_SYS_CDClose(SDL_CD *cdrom)
slouken@654
   442
{
slouken@654
   443
    close(cdrom->id);
slouken@654
   444
}
slouken@654
   445
slouken@654
   446
void SDL_SYS_CDQuit(void)
slouken@654
   447
{
slouken@654
   448
    int i;
slouken@654
   449
slouken@654
   450
    if ( SDL_numcds > 0 ) {
slouken@654
   451
	for ( i=0; i<SDL_numcds; ++i ) {
slouken@1336
   452
	    SDL_free(SDL_cdlist[i]);
slouken@654
   453
	}
slouken@654
   454
	SDL_numcds = 0;
slouken@654
   455
    }
slouken@654
   456
}
slouken@654
   457
slouken@874
   458