mame/src/emu/machine/scsicd.c

770 lines
18 KiB
C

/***************************************************************************
scsicd.c - Implementation of a SCSI CD-ROM device, using MAME's cdrom.c primitives
***************************************************************************/
#include "emu.h"
#include "machine/scsihle.h"
#include "cdrom.h"
#include "sound/cdda.h"
#include "imagedev/chd_cd.h"
#include "scsicd.h"
static void phys_frame_to_msf(int phys_frame, int *m, int *s, int *f)
{
*m = phys_frame / (60*75);
phys_frame -= (*m * 60 * 75);
*s = phys_frame / 75;
*f = phys_frame % 75;
}
// device type definition
const device_type SCSICD = &device_creator<scsicd_device>;
scsicd_device::scsicd_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock)
: scsihle_device(mconfig, SCSICD, "SCSICD", tag, owner, clock)
{
}
scsicd_device::scsicd_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, UINT32 clock) :
scsihle_device(mconfig, type, name, tag, owner, clock)
{
}
void scsicd_device::device_start()
{
scsihle_device::device_start();
save_item( NAME( lba ) );
save_item( NAME( blocks ) );
save_item( NAME( last_lba ) );
save_item( NAME( bytes_per_sector ) );
save_item( NAME( num_subblocks ) );
save_item( NAME( cur_subblock ) );
save_item( NAME( play_err_flag ) );
}
void scsicd_device::device_reset()
{
scsihle_device::device_reset();
is_file = TRUE;
cdrom = subdevice<cdrom_image_device>("image")->get_cdrom_file();
if( !cdrom )
{
// try to locate the CHD from a DISK_REGION
chd_file *chd = get_disk_handle( machine(), tag() );
if( chd != NULL )
{
is_file = FALSE;
cdrom = cdrom_open( chd );
}
}
if( !cdrom )
{
logerror( "SCSICD %s: no CD found!\n", tag() );
}
lba = 0;
blocks = 0;
last_lba = 0;
bytes_per_sector = 2048;
num_subblocks = 1;
cur_subblock = 0;
play_err_flag = 0;
}
void scsicd_device::device_stop()
{
if (!is_file)
{
if( cdrom )
{
cdrom_close( cdrom );
}
}
}
cdrom_interface scsicd_device::cd_intf = { 0, 0 };
static MACHINE_CONFIG_FRAGMENT(scsi_cdrom)
MCFG_CDROM_ADD("image", scsicd_device::cd_intf)
MACHINE_CONFIG_END
machine_config_constructor scsicd_device::device_mconfig_additions() const
{
return MACHINE_CONFIG_NAME(scsi_cdrom);
}
// scsicd_exec_command
//
// Execute a SCSI command.
void scsicd_device::ExecCommand( int *transferLength )
{
device_t *cdda;
int trk;
switch ( command[0] )
{
case 0x03: // REQUEST SENSE
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = SCSILengthFromUINT8( &command[ 4 ] );
break;
case 0x12: // INQUIRY
logerror("SCSICD: REQUEST SENSE\n");
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = SCSILengthFromUINT8( &command[ 4 ] );
break;
case 0x15: // MODE SELECT(6)
logerror("SCSICD: MODE SELECT(6) length %x control %x\n", command[4], command[5]);
SetPhase( SCSI_PHASE_DATAOUT );
*transferLength = SCSILengthFromUINT8( &command[ 4 ] );
break;
case 0x1a: // MODE SENSE(6)
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = SCSILengthFromUINT8( &command[ 4 ] );
break;
case 0x1b: // START STOP UNIT
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
{
cdda_stop_audio(cdda);
}
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0x1e: // PREVENT ALLOW MEDIUM REMOVAL
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0x25: // READ CAPACITY
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = 8;
break;
case 0x28: // READ(10)
lba = command[2]<<24 | command[3]<<16 | command[4]<<8 | command[5];
blocks = SCSILengthFromUINT16( &command[7] );
logerror("SCSICD: READ(10) at LBA %x for %d blocks (%d bytes)\n", lba, blocks, blocks * bytes_per_sector);
if (num_subblocks > 1)
{
cur_subblock = lba % num_subblocks;
lba /= num_subblocks;
}
else
{
cur_subblock = 0;
}
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
{
cdda_stop_audio(cdda);
}
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = blocks * bytes_per_sector;
break;
case 0x42: // READ SUB-CHANNEL
// logerror("SCSICD: READ SUB-CHANNEL type %d\n", command[3]);
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = SCSILengthFromUINT16( &command[ 7 ] );
break;
case 0x43: // READ TOC
{
int start_trk = command[6];
int end_trk = cdrom_get_last_track(cdrom);
int length;
int allocation_length = SCSILengthFromUINT16( &command[ 7 ] );
if( start_trk == 0 )
{
start_trk = 1;
}
if( start_trk == 0xaa )
{
end_trk = start_trk;
}
length = 4 + ( 8 * ( ( end_trk - start_trk ) + 1 ) );
if( length > allocation_length )
{
length = allocation_length;
}
else if( length < 4 )
{
length = 4;
}
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
{
cdda_stop_audio(cdda);
}
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = length;
break;
}
case 0x45: // PLAY AUDIO(10)
lba = command[2]<<24 | command[3]<<16 | command[4]<<8 | command[5];
blocks = SCSILengthFromUINT16( &command[7] );
// special cases: lba of 0 means MSF of 00:02:00
if (lba == 0)
{
lba = 150;
}
else if (lba == 0xffffffff)
{
logerror("SCSICD: play audio from current not implemented!\n");
}
logerror("SCSICD: PLAY AUDIO(10) at LBA %x for %x blocks\n", lba, blocks);
trk = cdrom_get_track(cdrom, lba);
if (cdrom_get_track_type(cdrom, trk) == CD_TRACK_AUDIO)
{
play_err_flag = 0;
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
cdda_start_audio(cdda, lba, blocks);
}
else
{
logerror("SCSICD: track is NOT audio!\n");
play_err_flag = 1;
}
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0x48: // PLAY AUDIO TRACK/INDEX
// be careful: tracks here are zero-based, but the SCSI command
// uses the real CD track number which is 1-based!
lba = cdrom_get_track_start(cdrom, command[4]-1);
blocks = cdrom_get_track_start(cdrom, command[7]-1) - lba;
if (command[4] > command[7])
{
blocks = 0;
}
if (command[4] == command[7])
{
blocks = cdrom_get_track_start(cdrom, command[4]) - lba;
}
if (blocks && cdrom)
{
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
cdda_start_audio(cdda, lba, blocks);
}
logerror("SCSICD: PLAY AUDIO T/I: strk %d idx %d etrk %d idx %d frames %d\n", command[4], command[5], command[7], command[8], blocks);
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0x4b: // PAUSE/RESUME
if (cdrom)
{
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
cdda_pause_audio(cdda, (command[8] & 0x01) ^ 0x01);
}
logerror("SCSICD: PAUSE/RESUME: %s\n", command[8]&1 ? "RESUME" : "PAUSE");
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0x55: // MODE SELECT(10)
logerror("SCSICD: MODE SELECT length %x control %x\n", command[7]<<8 | command[8], command[1]);
SetPhase( SCSI_PHASE_DATAOUT );
*transferLength = SCSILengthFromUINT16( &command[ 7 ] );
break;
case 0x5a: // MODE SENSE(10)
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = SCSILengthFromUINT16( &command[ 7 ] );
break;
case 0xa5: // PLAY AUDIO(12)
lba = command[2]<<24 | command[3]<<16 | command[4]<<8 | command[5];
blocks = command[6]<<24 | command[7]<<16 | command[8]<<8 | command[9];
// special cases: lba of 0 means MSF of 00:02:00
if (lba == 0)
{
lba = 150;
}
else if (lba == 0xffffffff)
{
logerror("SCSICD: play audio from current not implemented!\n");
}
logerror("SCSICD: PLAY AUDIO(12) at LBA %x for %x blocks\n", lba, blocks);
trk = cdrom_get_track(cdrom, lba);
if (cdrom_get_track_type(cdrom, trk) == CD_TRACK_AUDIO)
{
play_err_flag = 0;
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
cdda_start_audio(cdda, lba, blocks);
}
else
{
logerror("SCSICD: track is NOT audio!\n");
play_err_flag = 1;
}
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
case 0xa8: // READ(12)
lba = command[2]<<24 | command[3]<<16 | command[4]<<8 | command[5];
blocks = command[7]<<16 | command[8]<<8 | command[9];
logerror("SCSICD: READ(12) at LBA %x for %x blocks (%x bytes)\n", lba, blocks, blocks * bytes_per_sector);
if (num_subblocks > 1)
{
cur_subblock = lba % num_subblocks;
lba /= num_subblocks;
}
else
{
cur_subblock = 0;
}
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL)
{
cdda_stop_audio(cdda);
}
SetPhase( SCSI_PHASE_DATAIN );
*transferLength = blocks * bytes_per_sector;
break;
case 0xbb: // SET CD SPEED
logerror("SCSICD: SET CD SPEED to %d kbytes/sec.\n", command[2]<<8 | command[3]);
SetPhase( SCSI_PHASE_STATUS );
*transferLength = 0;
break;
default:
scsihle_device::ExecCommand( transferLength );
}
}
// scsicd_read_data
//
// Read data from the device resulting from the execution of a command
void scsicd_device::ReadData( UINT8 *data, int dataLength )
{
int i;
UINT32 last_phys_frame;
UINT32 temp;
UINT8 tmp_buffer[2048];
device_t *cdda;
switch ( command[0] )
{
case 0x03: // REQUEST SENSE
logerror("SCSICD: Reading REQUEST SENSE data\n");
memset( data, 0, dataLength );
data[0] = 0x71; // deferred error
cdda = cdda_from_cdrom(machine(), cdrom);
if (cdda != NULL && cdda_audio_active(cdda))
{
data[12] = 0x00;
data[13] = 0x11; // AUDIO PLAY OPERATION IN PROGRESS
}
else if (play_err_flag)
{
play_err_flag = 0;
data[12] = 0x64; // ILLEGAL MODE FOR THIS TRACK
data[13] = 0x00;
}
// (else 00/00 means no error to report)
break;
case 0x12: // INQUIRY
memset( data, 0, dataLength );
data[0] = 0x05; // device is present, device is CD/DVD (MMC-3)
data[1] = 0x80; // media is removable
data[2] = 0x05; // device complies with SPC-3 standard
data[3] = 0x02; // response data format = SPC-3 standard
// some Konami games freak out if this isn't "Sony", so we'll lie
// this is the actual drive on my Nagano '98 board
strcpy((char *)&data[8], "Sony");
strcpy((char *)&data[16], "CDU-76S");
strcpy((char *)&data[32], "1.0");
break;
case 0x25: // READ CAPACITY
logerror("SCSICD: READ CAPACITY\n");
temp = cdrom_get_track_start(cdrom, 0xaa);
temp--; // return the last used block on the disc
data[0] = (temp>>24) & 0xff;
data[1] = (temp>>16) & 0xff;
data[2] = (temp>>8) & 0xff;
data[3] = (temp & 0xff);
data[4] = 0;
data[5] = 0;
data[6] = (bytes_per_sector>>8)&0xff;
data[7] = (bytes_per_sector & 0xff);
break;
case 0x28: // READ(10)
case 0xa8: // READ(12)
logerror("SCSICD: read %x dataLength, \n", dataLength);
if ((cdrom) && (blocks))
{
while (dataLength > 0)
{
if (!cdrom_read_data(cdrom, lba, tmp_buffer, CD_TRACK_MODE1))
{
logerror("SCSICD: CD read error!\n");
}
logerror("True LBA: %d, buffer half: %d\n", lba, cur_subblock * bytes_per_sector);
memcpy(data, &tmp_buffer[cur_subblock * bytes_per_sector], bytes_per_sector);
cur_subblock++;
if (cur_subblock >= num_subblocks)
{
cur_subblock = 0;
lba++;
blocks--;
}
last_lba = lba;
dataLength -= bytes_per_sector;
data += bytes_per_sector;
}
}
break;
case 0x42: // READ SUB-CHANNEL
switch (command[3])
{
case 1: // return current position
{
int audio_active;
int msf;
if (!cdrom)
{
return;
}
logerror("SCSICD: READ SUB-CHANNEL Time = %x, SUBQ = %x\n", command[1], command[2]);
msf = command[1] & 0x2;
cdda = cdda_from_cdrom(machine(), cdrom);
audio_active = cdda != NULL && cdda_audio_active(cdda);
if (audio_active)
{
if (cdda_audio_paused(cdda))
{
data[1] = 0x12; // audio is paused
}
else
{
data[1] = 0x11; // audio in progress
}
}
else
{
if (cdda != NULL && cdda_audio_ended(cdda))
{
data[1] = 0x13; // ended successfully
}
else
{
// data[1] = 0x14; // stopped due to error
data[1] = 0x15; // No current audio status to return
}
}
// if audio is playing, get the latest LBA from the CDROM layer
if (audio_active)
{
last_lba = cdda_get_audio_lba(cdda);
}
else
{
last_lba = 0;
}
data[2] = 0;
data[3] = 12; // data length
data[4] = 0x01; // sub-channel format code
data[5] = 0x10 | (audio_active ? 0 : 4);
data[6] = cdrom_get_track(cdrom, last_lba) + 1; // track
data[7] = 0; // index
last_phys_frame = last_lba;
if (msf)
{
int m,s,f;
phys_frame_to_msf(last_phys_frame, &m, &s, &f);
data[8] = 0;
data[9] = m;
data[10] = s;
data[11] = f;
}
else
{
data[8] = last_phys_frame>>24;
data[9] = (last_phys_frame>>16)&0xff;
data[10] = (last_phys_frame>>8)&0xff;
data[11] = last_phys_frame&0xff;
}
last_phys_frame -= cdrom_get_track_start(cdrom, data[6] - 1);
if (msf)
{
int m,s,f;
phys_frame_to_msf(last_phys_frame, &m, &s, &f);
data[12] = 0;
data[13] = m;
data[14] = s;
data[15] = f;
}
else
{
data[12] = last_phys_frame>>24;
data[13] = (last_phys_frame>>16)&0xff;
data[14] = (last_phys_frame>>8)&0xff;
data[15] = last_phys_frame&0xff;
}
break;
}
default:
logerror("SCSICD: Unknown subchannel type %d requested\n", command[3]);
}
break;
case 0x43: // READ TOC
/*
Track numbers are problematic here: 0 = lead-in, 0xaa = lead-out.
That makes sense in terms of how real-world CDs are referred to, but
our internal routines for tracks use "0" as track 1. That probably
should be fixed...
*/
logerror("SCSICD: READ TOC, format = %d time=%d\n", command[2]&0xf,(command[1]>>1)&1);
switch (command[2] & 0x0f)
{
case 0: // normal
{
int start_trk;
int end_trk;
int len;
int in_len;
int dptr;
UINT32 tstart;
start_trk = command[6];
if( start_trk == 0 )
{
start_trk = 1;
}
end_trk = cdrom_get_last_track(cdrom);
len = (end_trk * 8) + 2;
// the returned TOC DATA LENGTH must be the full amount,
// regardless of how much we're able to pass back due to in_len
dptr = 0;
data[dptr++] = (len>>8) & 0xff;
data[dptr++] = (len & 0xff);
data[dptr++] = 1;
data[dptr++] = end_trk;
if( start_trk == 0xaa )
{
end_trk = 0xaa;
}
in_len = command[7]<<8 | command[8];
for (i = start_trk; i <= end_trk; i++)
{
int cdrom_track = i;
if( cdrom_track != 0xaa )
{
cdrom_track--;
}
if( dptr >= in_len )
{
break;
}
data[dptr++] = 0;
data[dptr++] = cdrom_get_adr_control(cdrom, cdrom_track);
data[dptr++] = i;
data[dptr++] = 0;
tstart = cdrom_get_track_start(cdrom, cdrom_track);
if ((command[1]&2)>>1)
tstart = lba_to_msf(tstart);
data[dptr++] = (tstart>>24) & 0xff;
data[dptr++] = (tstart>>16) & 0xff;
data[dptr++] = (tstart>>8) & 0xff;
data[dptr++] = (tstart & 0xff);
}
}
break;
default:
logerror("SCSICD: Unhandled READ TOC format %d\n", command[2]&0xf);
break;
}
break;
case 0x1a: // MODE SENSE(6)
case 0x5a: // MODE SENSE(10)
logerror("SCSICD: MODE SENSE page code = %x, PC = %x\n", command[2] & 0x3f, (command[2]&0xc0)>>6);
switch (command[2] & 0x3f)
{
case 0xe: // CD Audio control page
data[0] = 0x8e; // page E, parameter is savable
data[1] = 0x0e; // page length
data[2] = 0x04; // IMMED = 1, SOTC = 0
data[3] = data[4] = data[5] = data[6] = data[7] = 0; // reserved
// connect each audio channel to 1 output port
data[8] = 1;
data[10] = 2;
data[12] = 4;
data[14] = 8;
// indicate max volume
data[9] = data[11] = data[13] = data[15] = 0xff;
break;
case 0x2a: // Page capabilities
data[0] = 0x2a;
data[1] = 0x14; // page length
data[2] = 0x00; data[3] = 0x00; // CD-R only
data[4] = 0x01; // can play audio
data[5] = 0;
data[6] = 0;
data[7] = 0;
data[8] = 0x02; data[9] = 0xc0; // 4x speed
data[10] = 0;
data[11] = 2; // two volumen levels
data[12] = 0x00; data[13] = 0x00; // buffer
data[14] = 0x02; data[15] = 0xc0; // 4x read speed
data[16] = 0;
data[17] = 0;
data[18] = 0;
data[19] = 0;
data[20] = 0;
data[21] = 0;
break;
default:
logerror("SCSICD: MODE SENSE unknown page %x\n", command[2] & 0x3f);
break;
}
break;
default:
scsihle_device::ReadData( data, dataLength );
break;
}
}
// scsicd_write_data
//
// Write data to the CD-ROM device as part of the execution of a command
void scsicd_device::WriteData( UINT8 *data, int dataLength )
{
switch (command[ 0 ])
{
case 0x15: // MODE SELECT(6)
case 0x55: // MODE SELECT(10)
logerror("SCSICD: MODE SELECT page %x\n", data[0] & 0x3f);
switch (data[0] & 0x3f)
{
case 0x0: // vendor-specific
// check for SGI extension to force 512-byte blocks
if ((data[3] == 8) && (data[10] == 2))
{
logerror("SCSICD: Experimental SGI 512-byte block extension enabled\n");
bytes_per_sector = 512;
num_subblocks = 4;
}
else
{
logerror("SCSICD: Unknown vendor-specific page!\n");
}
break;
case 0xe: // audio page
logerror("Ch 0 route: %x vol: %x\n", data[8], data[9]);
logerror("Ch 1 route: %x vol: %x\n", data[10], data[11]);
logerror("Ch 2 route: %x vol: %x\n", data[12], data[13]);
logerror("Ch 3 route: %x vol: %x\n", data[14], data[15]);
break;
}
break;
default:
scsihle_device::WriteData( data, dataLength );
break;
}
}
void scsicd_device::GetDevice( void **_cdrom )
{
*(cdrom_file **)_cdrom = cdrom;
}
void scsicd_device::SetDevice( void *_cdrom )
{
cdrom = (cdrom_file *)_cdrom;
}
int scsicd_device::GetSectorBytes()
{
return bytes_per_sector;
}