mirror of
https://git.do.srb2.org/KartKrew/Kart-Public.git
synced 2025-01-02 15:51:12 +00:00
756 lines
19 KiB
C
756 lines
19 KiB
C
|
/* bcd.c -- Brennan's CD-ROM Audio Playing Library
|
||
|
by Brennan Underwood, http://brennan.home.ml.org/ */
|
||
|
#include <dos.h>
|
||
|
#include <dpmi.h>
|
||
|
#include <go32.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <stdio.h>
|
||
|
#include <malloc.h>
|
||
|
#include <unistd.h>
|
||
|
#include <strings.h>
|
||
|
#ifdef STANDALONE
|
||
|
#include <conio.h> /* for getch() */
|
||
|
#endif
|
||
|
|
||
|
#include "bcd.h"
|
||
|
|
||
|
typedef struct {
|
||
|
int is_audio;
|
||
|
int start, end, len;
|
||
|
} Track;
|
||
|
|
||
|
static int mscdex_version;
|
||
|
static int first_drive;
|
||
|
static int num_tracks;
|
||
|
static int lowest_track, highest_track;
|
||
|
static int audio_length;
|
||
|
|
||
|
#ifdef STATIC_TRACKS
|
||
|
static Track tracks[99];
|
||
|
#else
|
||
|
static Track *tracks;
|
||
|
#endif
|
||
|
|
||
|
static int dos_mem_segment, dos_mem_selector = -1;
|
||
|
|
||
|
int _status, _error, _error_code;
|
||
|
const char *_bcd_error = NULL;
|
||
|
|
||
|
#define RESET_ERROR (_error = _error_code = 0)
|
||
|
#define ERROR_BIT (1 << 15)
|
||
|
#define BUSY_BIT (1 << 9)
|
||
|
#ifndef TRUE
|
||
|
#define TRUE 1
|
||
|
#define FALSE 0
|
||
|
#endif
|
||
|
|
||
|
#pragma pack(1)
|
||
|
|
||
|
/* I know 'typedef struct {} bleh' is a bad habit, but... */
|
||
|
typedef struct {
|
||
|
unsigned char len;
|
||
|
unsigned char unit;
|
||
|
unsigned char command;
|
||
|
unsigned short status;
|
||
|
unsigned char reserved[8];
|
||
|
} ATTRPACK RequestHeader;
|
||
|
|
||
|
typedef struct {
|
||
|
RequestHeader request_header;
|
||
|
unsigned char descriptor;
|
||
|
unsigned long address;
|
||
|
unsigned short len;
|
||
|
unsigned short secnum;
|
||
|
unsigned long ptr;
|
||
|
} ATTRPACK IOCTLI;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char lowest;
|
||
|
unsigned char highest;
|
||
|
char total[4];
|
||
|
} ATTRPACK DiskInfo;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char track_number;
|
||
|
char start[4];
|
||
|
unsigned char info;
|
||
|
} ATTRPACK TrackInfo;
|
||
|
|
||
|
typedef struct {
|
||
|
RequestHeader request;
|
||
|
unsigned char mode;
|
||
|
unsigned long start;
|
||
|
unsigned long len;
|
||
|
} ATTRPACK PlayRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
RequestHeader request;
|
||
|
} ATTRPACK StopRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
RequestHeader request;
|
||
|
} ATTRPACK ResumeRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char input0;
|
||
|
unsigned char volume0;
|
||
|
unsigned char input1;
|
||
|
unsigned char volume1;
|
||
|
unsigned char input2;
|
||
|
unsigned char volume2;
|
||
|
unsigned char input3;
|
||
|
unsigned char volume3;
|
||
|
} ATTRPACK VolumeRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char fn;
|
||
|
} ATTRPACK LockRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char mbyte;
|
||
|
} ATTRPACK MediaChangedRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned long status;
|
||
|
} ATTRPACK StatusRequest;
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char control;
|
||
|
unsigned char mode;
|
||
|
unsigned long loc;
|
||
|
} ATTRPACK PositionRequest;
|
||
|
|
||
|
#pragma pack()
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
const char *bcd_error(void) {
|
||
|
static char retstr[132];
|
||
|
const char *errorcodes[] = {
|
||
|
"Write-protect violation",
|
||
|
"Unknown unit",
|
||
|
"Drive not ready",
|
||
|
"Unknown command",
|
||
|
"CRC error",
|
||
|
"Bad drive request structure length",
|
||
|
"Seek error",
|
||
|
"Unknown media",
|
||
|
"Sector not found",
|
||
|
"Printer out of paper: world coming to an end",/* I mean really, on a CD? */
|
||
|
"Write fault",
|
||
|
"Read fault",
|
||
|
"General failure",
|
||
|
"Reserved",
|
||
|
"Reserved",
|
||
|
"Invalid disk change"
|
||
|
};
|
||
|
*retstr = 0;
|
||
|
if (_error != 0) {
|
||
|
strcat(retstr, "Device error: ");
|
||
|
if (_error_code < 0 || _error_code > 0xf)
|
||
|
strcat(retstr, "Invalid error");
|
||
|
else
|
||
|
strcat(retstr, errorcodes[_error_code]);
|
||
|
strcat(retstr, " ");
|
||
|
}
|
||
|
if (_bcd_error != NULL) {
|
||
|
if (*retstr) strcat(retstr, ", ");
|
||
|
strcat(retstr, "BCD error: ");
|
||
|
strcat(retstr, _bcd_error);
|
||
|
}
|
||
|
return retstr;
|
||
|
}
|
||
|
|
||
|
/* DOS IOCTL w/ command block */
|
||
|
static void bcd_ioctl(IOCTLI *ioctli, void *command, int len) {
|
||
|
int ioctli_len = sizeof (IOCTLI);
|
||
|
unsigned long command_address = dos_mem_segment << 4;
|
||
|
__dpmi_regs regs;
|
||
|
|
||
|
memset(®s, 0, sizeof regs);
|
||
|
regs.x.es = (__tb >> 4) & 0xffff;
|
||
|
regs.x.ax = 0x1510;
|
||
|
regs.x.bx = __tb & 0xf;
|
||
|
regs.x.cx = first_drive;
|
||
|
ioctli->address = dos_mem_segment << 16;
|
||
|
ioctli->len = len;
|
||
|
dosmemput(ioctli, ioctli_len, __tb); /* put ioctl into dos area */
|
||
|
dosmemput(command, len, command_address); /* and command too */
|
||
|
if (__dpmi_int(0x2f, ®s) == -1) {
|
||
|
_bcd_error = "__dpmi_int() failed";
|
||
|
return;
|
||
|
}
|
||
|
dosmemget(__tb, ioctli_len, ioctli); /* retrieve results */
|
||
|
dosmemget(command_address, len, command);
|
||
|
_status = ioctli->request_header.status;
|
||
|
if (_status & ERROR_BIT) {
|
||
|
_error = TRUE;
|
||
|
_error_code = _status & 0xff;
|
||
|
} else {
|
||
|
_error = FALSE;
|
||
|
_error_code = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* no command block */
|
||
|
FUNCINLINE static ATTRINLINE void bcd_ioctl2(void *cmd, int len) {
|
||
|
__dpmi_regs regs;
|
||
|
memset(®s, 0, sizeof regs);
|
||
|
regs.x.es = (__tb >> 4) & 0xffff;
|
||
|
regs.x.ax = 0x1510;
|
||
|
regs.x.bx = __tb & 0xf;
|
||
|
regs.x.cx = first_drive;
|
||
|
dosmemput(cmd, len, __tb); /* put ioctl block in dos arena */
|
||
|
if (__dpmi_int(0x2f, ®s) == -1) {
|
||
|
_bcd_error = "__dpmi_int() failed";
|
||
|
return;
|
||
|
}
|
||
|
/* I hate to have no error capability for ioctl2 but the command block
|
||
|
doesn't necessarily have a status field */
|
||
|
RESET_ERROR;
|
||
|
}
|
||
|
|
||
|
FUNCINLINE static ATTRINLINE int red2hsg(char *r) {
|
||
|
return r[0] + r[1]*75 + r[2]*4500 - 150;
|
||
|
}
|
||
|
|
||
|
int bcd_now_playing(void) {
|
||
|
int i, loc = bcd_audio_position();
|
||
|
_bcd_error = NULL;
|
||
|
if (!bcd_audio_busy()) {
|
||
|
_bcd_error = "Audio not playing";
|
||
|
return 0;
|
||
|
}
|
||
|
if (
|
||
|
#ifndef STATIC_TRACKS
|
||
|
tracks == NULL &&
|
||
|
#endif
|
||
|
!bcd_get_audio_info())
|
||
|
return 0;
|
||
|
for (i = lowest_track; i <= highest_track; i++) {
|
||
|
if (loc >= tracks[i].start && loc <= tracks[i].end) return i;
|
||
|
}
|
||
|
/* some bizarre location? */
|
||
|
_bcd_error = "Head outside of bounds";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* handles the setup for CD-ROM audio interface */
|
||
|
int bcd_open(void) {
|
||
|
__dpmi_regs regs;
|
||
|
_bcd_error = NULL;
|
||
|
|
||
|
/* disk I/O wouldn't work anyway if you set sizeof tb this low, but... */
|
||
|
if (_go32_info_block.size_of_transfer_buffer < 4096) {
|
||
|
_bcd_error = "Transfer buffer too small";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
memset(®s, 0, sizeof regs);
|
||
|
regs.x.ax = 0x1500;
|
||
|
regs.x.bx = 0x0;
|
||
|
__dpmi_int(0x2f, ®s);
|
||
|
if (regs.x.bx == 0) { /* abba no longer lives */
|
||
|
_bcd_error = "MSCDEX not found";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
first_drive = regs.x.cx; /* use the first drive */
|
||
|
|
||
|
/* check for mscdex at least 2.0 */
|
||
|
memset(®s, 0, sizeof regs);
|
||
|
regs.x.ax = 0x150C;
|
||
|
__dpmi_int(0x2f, ®s);
|
||
|
if (regs.x.bx == 0) {
|
||
|
_bcd_error = "MSCDEX version < 2.0";
|
||
|
return 0;
|
||
|
}
|
||
|
mscdex_version = regs.x.bx;
|
||
|
|
||
|
/* allocate 256 bytes of dos memory for the command blocks */
|
||
|
if ((dos_mem_segment = __dpmi_allocate_dos_memory(16, &dos_mem_selector))<0) {
|
||
|
_bcd_error = "Could not allocate 256 bytes of DOS memory";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return mscdex_version;
|
||
|
}
|
||
|
|
||
|
/* Shuts down CD-ROM audio interface */
|
||
|
int bcd_close(void) {
|
||
|
_bcd_error = NULL;
|
||
|
if (dos_mem_selector != -1) {
|
||
|
__dpmi_free_dos_memory(dos_mem_selector);
|
||
|
dos_mem_selector = -1;
|
||
|
}
|
||
|
#ifndef STATIC_TRACKS
|
||
|
if (tracks) free(tracks);
|
||
|
tracks = NULL;
|
||
|
#endif
|
||
|
RESET_ERROR;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_open_door(void) {
|
||
|
IOCTLI ioctli;
|
||
|
char eject = 0;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 12;
|
||
|
ioctli.len = 1;
|
||
|
bcd_ioctl(&ioctli, &eject, sizeof eject);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_close_door(void) {
|
||
|
IOCTLI ioctli;
|
||
|
char closeit = 5;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 12;
|
||
|
ioctli.len = 1;
|
||
|
bcd_ioctl(&ioctli, &closeit, sizeof closeit);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_lock(int fn) {
|
||
|
IOCTLI ioctli;
|
||
|
LockRequest req;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
memset(&req, 0, sizeof req);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 12;
|
||
|
ioctli.len = sizeof req;
|
||
|
req.control = 1;
|
||
|
req.fn = fn ? 1 : 0;
|
||
|
bcd_ioctl(&ioctli, &req, sizeof req);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
int bcd_disc_changed(void) {
|
||
|
IOCTLI ioctli;
|
||
|
MediaChangedRequest req;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
memset(&req, 0, sizeof req);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 3;
|
||
|
ioctli.len = sizeof req;
|
||
|
req.control = 9;
|
||
|
bcd_ioctl(&ioctli, &req, sizeof req);
|
||
|
return req.mbyte;
|
||
|
}
|
||
|
|
||
|
int bcd_reset(void) {
|
||
|
IOCTLI ioctli;
|
||
|
char reset = 2;
|
||
|
_bcd_error = NULL;
|
||
|
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 12;
|
||
|
ioctli.len = 1;
|
||
|
bcd_ioctl(&ioctli, &reset, sizeof reset);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_device_status(void) {
|
||
|
IOCTLI ioctli;
|
||
|
StatusRequest req;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
memset(&req, 0, sizeof req);
|
||
|
ioctli.request_header.len = sizeof ioctli; // ok
|
||
|
ioctli.request_header.command = 3;
|
||
|
ioctli.len = sizeof req;
|
||
|
req.control = 6;
|
||
|
bcd_ioctl(&ioctli, &req, sizeof req);
|
||
|
return req.status;
|
||
|
}
|
||
|
|
||
|
static inline int bcd_get_status_word(void) {
|
||
|
IOCTLI ioctli;
|
||
|
DiskInfo disk_info;
|
||
|
|
||
|
/* get cd info as an excuse to get a look at the status word */
|
||
|
memset(&disk_info, 0, sizeof disk_info);
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
|
||
|
ioctli.request_header.len = 26;
|
||
|
ioctli.request_header.command = 3;
|
||
|
ioctli.len = 7;
|
||
|
disk_info.control = 10;
|
||
|
bcd_ioctl(&ioctli, &disk_info, sizeof disk_info);
|
||
|
return _status;
|
||
|
}
|
||
|
|
||
|
int bcd_audio_busy(void) {
|
||
|
_bcd_error = NULL;
|
||
|
/* If the door is open, then the head is busy, and so the busy bit is
|
||
|
on. It is not, however, playing audio. */
|
||
|
if (bcd_device_status() & BCD_DOOR_OPEN)
|
||
|
return 0;
|
||
|
|
||
|
bcd_get_status_word();
|
||
|
if (_error) return -1;
|
||
|
return (_status & BUSY_BIT) ? 1 : 0;
|
||
|
}
|
||
|
|
||
|
int bcd_audio_position(void) {
|
||
|
IOCTLI ioctli;
|
||
|
PositionRequest req;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
memset(&req, 0, sizeof req);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 3;
|
||
|
ioctli.len = sizeof req;
|
||
|
req.control = 1;
|
||
|
bcd_ioctl(&ioctli, &req, sizeof req);
|
||
|
return req.loc;
|
||
|
}
|
||
|
|
||
|
/* Internal function to get track info */
|
||
|
static inline void bcd_get_track_info(int n, Track *t) {
|
||
|
IOCTLI ioctli;
|
||
|
TrackInfo info;
|
||
|
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
memset(&info, 0, sizeof info);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 3;
|
||
|
info.control = 11;
|
||
|
info.track_number = n;
|
||
|
bcd_ioctl(&ioctli, &info, sizeof info);
|
||
|
t->start = red2hsg(info.start);
|
||
|
if (info.info & 64)
|
||
|
t->is_audio = 0;
|
||
|
else
|
||
|
t->is_audio = 1;
|
||
|
}
|
||
|
|
||
|
int bcd_get_audio_info(void) {
|
||
|
IOCTLI ioctli;
|
||
|
DiskInfo disk_info;
|
||
|
int i;
|
||
|
|
||
|
|
||
|
_bcd_error = NULL;
|
||
|
#ifndef STATIC_TRACKS
|
||
|
if (tracks) free(tracks);
|
||
|
tracks = NULL;
|
||
|
#endif
|
||
|
|
||
|
memset(&disk_info, 0, sizeof disk_info);
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
|
||
|
ioctli.request_header.len = 26;
|
||
|
ioctli.request_header.command = 3;
|
||
|
ioctli.len = 7;
|
||
|
disk_info.control = 10;
|
||
|
bcd_ioctl(&ioctli, &disk_info, sizeof disk_info);
|
||
|
if (_error) return 0;
|
||
|
|
||
|
lowest_track = disk_info.lowest;
|
||
|
highest_track = disk_info.highest;
|
||
|
num_tracks = disk_info.highest - disk_info.lowest + 1;
|
||
|
|
||
|
#ifndef STATIC_TRACKS
|
||
|
//tracks = calloc(num_tracks, sizeof (Track));
|
||
|
/* alloc max space in order to attempt to avoid possible overrun bug */
|
||
|
tracks = calloc(highest_track+1, sizeof (Track));
|
||
|
if (tracks == NULL) {
|
||
|
_bcd_error = "Out of memory allocating tracks\n";
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* get track starts */
|
||
|
for (i = lowest_track; i <= highest_track; i++)
|
||
|
bcd_get_track_info(i, tracks+i);
|
||
|
|
||
|
/* figure out track ends */
|
||
|
for (i = lowest_track; i < highest_track; i++)
|
||
|
tracks[i].end = tracks[i+1].start-1;
|
||
|
audio_length = red2hsg(disk_info.total);
|
||
|
tracks[i].end = audio_length;
|
||
|
for (i = lowest_track; i <= highest_track; i++)
|
||
|
tracks[i].len = tracks[i].end - tracks[i].start;
|
||
|
|
||
|
return num_tracks;
|
||
|
}
|
||
|
|
||
|
int bcd_get_track_address(int trackno, int *start, int *len) {
|
||
|
_bcd_error = NULL;
|
||
|
//if (trackno >= num_tracks+1 || trackno <= 0) {
|
||
|
if (trackno < lowest_track || trackno > highest_track) {
|
||
|
_bcd_error = "Track out of range";
|
||
|
*start = *len = 0;
|
||
|
return 0;
|
||
|
}
|
||
|
*start = tracks[trackno].start;
|
||
|
*len = tracks[trackno].len;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_track_is_audio(int trackno) {
|
||
|
//if (trackno >= num_tracks+1 || trackno <= 0) {
|
||
|
if (trackno < lowest_track || trackno > highest_track) {
|
||
|
_bcd_error = "Track out of range";
|
||
|
return 0;
|
||
|
}
|
||
|
return tracks[trackno].is_audio;
|
||
|
}
|
||
|
|
||
|
int bcd_set_volume(int volume) {
|
||
|
IOCTLI ioctli;
|
||
|
VolumeRequest v;
|
||
|
|
||
|
_bcd_error = NULL;
|
||
|
if (volume > 255) volume = 255;
|
||
|
else if (volume < 0) volume = 0;
|
||
|
memset(&ioctli, 0, sizeof ioctli);
|
||
|
ioctli.request_header.len = sizeof ioctli;
|
||
|
ioctli.request_header.command = 12;
|
||
|
ioctli.len = sizeof v;
|
||
|
v.control = 3;
|
||
|
v.volume0 = volume;
|
||
|
v.input0 = 0;
|
||
|
v.volume1 = volume;
|
||
|
v.input1 = 1;
|
||
|
v.volume2 = volume;
|
||
|
v.input2 = 2;
|
||
|
v.volume3 = volume;
|
||
|
v.input3 = 3;
|
||
|
|
||
|
bcd_ioctl(&ioctli, &v, sizeof v);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_play(int location, int frames) {
|
||
|
PlayRequest cmd;
|
||
|
memset(&cmd, 0, sizeof cmd);
|
||
|
|
||
|
_bcd_error = NULL;
|
||
|
/* the following should be in user code, but it'll fail otherwise */
|
||
|
if (bcd_audio_busy())
|
||
|
bcd_stop();
|
||
|
|
||
|
cmd.request.len = sizeof cmd;
|
||
|
cmd.request.command = 132;
|
||
|
cmd.start = location;
|
||
|
cmd.len = frames;
|
||
|
bcd_ioctl2(&cmd, sizeof cmd);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_play_track(int trackno) {
|
||
|
_bcd_error = NULL;
|
||
|
if (!bcd_get_audio_info()) return 0;
|
||
|
|
||
|
if (trackno < lowest_track || trackno > highest_track) {
|
||
|
_bcd_error = "Track out of range";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (! tracks[trackno].is_audio) {
|
||
|
_bcd_error = "Not an audio track";
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return bcd_play(tracks[trackno].start, tracks[trackno].len);
|
||
|
}
|
||
|
|
||
|
int bcd_stop(void) {
|
||
|
StopRequest cmd;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&cmd, 0, sizeof cmd);
|
||
|
cmd.request.len = sizeof cmd;
|
||
|
cmd.request.command = 133;
|
||
|
bcd_ioctl2(&cmd, sizeof cmd);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int bcd_resume(void) {
|
||
|
ResumeRequest cmd;
|
||
|
_bcd_error = NULL;
|
||
|
memset(&cmd, 0, sizeof cmd);
|
||
|
cmd.request.len = sizeof cmd;
|
||
|
cmd.request.command = 136;
|
||
|
bcd_ioctl2(&cmd, sizeof cmd);
|
||
|
if (_error) return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#ifdef STANDALONE
|
||
|
static char *card(int c) {
|
||
|
return c == 1 ? "" : "s";
|
||
|
}
|
||
|
|
||
|
static void print_hsg(int hsg) {
|
||
|
int hours, minutes, seconds;
|
||
|
seconds = hsg / 75;
|
||
|
minutes = seconds / 60;
|
||
|
seconds %= 60;
|
||
|
hours = minutes / 60;
|
||
|
minutes %= 60;
|
||
|
printf("%2d:%02d:%02d", hours, minutes, seconds);
|
||
|
}
|
||
|
|
||
|
static void print_binary(int v, int len) {
|
||
|
for (;len;len--)
|
||
|
printf("%d", (v & (1 << len)) ? 1 : 0);
|
||
|
}
|
||
|
|
||
|
int main(int argc, char *argv[]) {
|
||
|
int i, n1, n2, t;
|
||
|
|
||
|
if (!bcd_open()) {
|
||
|
fprintf(stderr, "Couldn't open CD-ROM drive. %s\n", bcd_error());
|
||
|
exit(0);
|
||
|
}
|
||
|
|
||
|
for (i = 1; i < argc; i++) {
|
||
|
if (*argv[i] == '-') strcpy(argv[i], argv[i]+1);
|
||
|
if (!strcmp(argv[i], "open") || !strcmp(argv[i], "eject")) {
|
||
|
bcd_open_door();
|
||
|
} else if (!strcmp(argv[i], "close")) {
|
||
|
bcd_close_door();
|
||
|
} else if (!strcmp(argv[i], "sleep")) {
|
||
|
if (++i >= argc) break;
|
||
|
sleep(atoi(argv[i]));
|
||
|
} else if (!strcmp(argv[i], "list")) {
|
||
|
int nd = 0, na = 0, audio_time = 0;
|
||
|
if (!bcd_get_audio_info()) {
|
||
|
printf("Error getting audio info\n");
|
||
|
} else if (lowest_track == 0) {
|
||
|
printf("No audio tracks\n");
|
||
|
} else {
|
||
|
for (t = lowest_track; t <= highest_track; t++) {
|
||
|
printf("Track %2d: ", t);
|
||
|
print_hsg(tracks[t].start);
|
||
|
printf(" -> ");
|
||
|
print_hsg(tracks[t].end);
|
||
|
printf(" (");
|
||
|
print_hsg(tracks[t].len);
|
||
|
if (tracks[t].is_audio) {
|
||
|
na++;
|
||
|
printf(") audio");
|
||
|
audio_time += tracks[t].len;
|
||
|
} else {
|
||
|
nd++;
|
||
|
printf(") data ");
|
||
|
}
|
||
|
printf(" (HSG: %06d->%06d)\n", tracks[t].start, tracks[t].end);
|
||
|
}
|
||
|
printf("%d audio track%s, %d data track%s\n", na, card(na), nd, card(nd));
|
||
|
if (audio_time) {
|
||
|
printf("Audio time: ");
|
||
|
print_hsg(audio_time);
|
||
|
printf("\n");
|
||
|
}
|
||
|
}
|
||
|
} else if (!strcmp(argv[i], "lock")) {
|
||
|
bcd_lock(1);
|
||
|
} else if (!strcmp(argv[i], "pladdr")) {
|
||
|
if (++i >= argc) break;
|
||
|
n1 = atoi(argv[i]);
|
||
|
if (++i >= argc) break;
|
||
|
n2 = atoi(argv[i]);
|
||
|
printf("Playing frame %d to frame %d\n", n1, n2);
|
||
|
bcd_play(n1, n2-n1);
|
||
|
} else if (!strcmp(argv[i], "play")) {
|
||
|
if (++i >= argc) break;
|
||
|
if (bcd_audio_busy()) {
|
||
|
bcd_stop();
|
||
|
delay(1000);
|
||
|
}
|
||
|
n1 = atoi(argv[i]);
|
||
|
printf("Playing track %d\n", n1);
|
||
|
bcd_play_track(n1);
|
||
|
} else if (!strcmp(argv[i], "reset")) {
|
||
|
bcd_reset();
|
||
|
} else if (!strcmp(argv[i], "resume")) {
|
||
|
bcd_resume();
|
||
|
} else if (!strcmp(argv[i], "status")) {
|
||
|
int s;
|
||
|
s = bcd_device_status();
|
||
|
printf("MSCDEX version %d.%d\n", mscdex_version >> 8,
|
||
|
mscdex_version & 0xff);
|
||
|
printf("Device status word '");
|
||
|
print_binary(s, 16);
|
||
|
printf("'\nDoor is %sopen\n", s & BCD_DOOR_OPEN ? "" : "not ");
|
||
|
printf("Door is %slocked\n", s & BCD_DOOR_UNLOCKED ? "not " : "");
|
||
|
printf("Audio is %sbusy\n", bcd_audio_busy() ? "" : "not ");
|
||
|
s = bcd_disc_changed();
|
||
|
if (s == BCD_DISC_UNKNOWN) printf("Media change status unknown\n");
|
||
|
else printf("Media has %schanged\n",
|
||
|
(s == BCD_DISC_CHANGED) ? "" : "not ");
|
||
|
} else if (!strcmp(argv[i], "stop")) {
|
||
|
bcd_stop();
|
||
|
} else if (!strcmp(argv[i], "unlock")) {
|
||
|
bcd_lock(0);
|
||
|
} else if (!strcmp(argv[i], "volume")) {
|
||
|
bcd_set_volume(atoi(argv[++i]));
|
||
|
} else if (!strcmp(argv[i], "wait")) {
|
||
|
while (bcd_audio_busy()) {
|
||
|
int n = bcd_now_playing();
|
||
|
if (n == 0) break;
|
||
|
printf("%2d: ", n);
|
||
|
print_hsg(bcd_audio_position() - tracks[n].start);
|
||
|
printf("\r");
|
||
|
fflush(stdout);
|
||
|
delay(100);
|
||
|
if (kbhit() && getch() == 27) break;
|
||
|
}
|
||
|
printf("\n");
|
||
|
} else if (!strcmp(argv[i], "help") || !strcmp(argv[i], "usage")) {
|
||
|
printf("BCD version %x.%x\n" \
|
||
|
"Usage: BCD {commands}\n" \
|
||
|
"Valid commands:\n" \
|
||
|
"\tclose - close door/tray\n" \
|
||
|
"\tdelay {n} - delay {n} seconds\n" \
|
||
|
"\tlist - list track info\n" \
|
||
|
"\tlock - lock door/tray\n" \
|
||
|
"\topen - open door/tray\n" \
|
||
|
"\tpladdr {n1} {n2}- play frame {n1} to {n2}\n" \
|
||
|
"\tplay {n} - play track {n}\n" \
|
||
|
"\treset - reset the drive\n" \
|
||
|
"\tresume - resume from last stop\n" \
|
||
|
"\tstatus - show drive status\n" \
|
||
|
"\tstop - stop audio playback\n" \
|
||
|
"\tunlock - unlock door/tray\n" \
|
||
|
"\tvolume {n} - set volume to {n} where 0 <= {n} <= 255\n",
|
||
|
BCD_VERSION >> 8, BCD_VERSION & 0xff);
|
||
|
} else
|
||
|
printf("Unknown command '%s'\n", argv[i]);
|
||
|
if (_error || _bcd_error) printf("%s\n", bcd_error());
|
||
|
}
|
||
|
bcd_close();
|
||
|
exit(0);
|
||
|
}
|
||
|
#endif
|