2011-05-14 21:56:54 +00:00
|
|
|
|
using System;
|
2011-08-02 08:26:33 +00:00
|
|
|
|
using System.Collections.Generic;
|
2011-05-14 21:56:54 +00:00
|
|
|
|
|
|
|
|
|
//main apis for emulator core routine use
|
|
|
|
|
|
2011-08-03 00:57:01 +00:00
|
|
|
|
namespace BizHawk.DiscSystem
|
2011-05-14 21:56:54 +00:00
|
|
|
|
{
|
2011-08-06 10:43:05 +00:00
|
|
|
|
public class DiscReferenceException : Exception
|
|
|
|
|
{
|
|
|
|
|
public DiscReferenceException(string fname, Exception inner)
|
|
|
|
|
: base(string.Format("A disc attempted to reference a file which could not be accessed or loaded: {0}", fname),inner)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-02 08:26:33 +00:00
|
|
|
|
public class DiscHopper
|
|
|
|
|
{
|
|
|
|
|
public Disc CurrentDisc;
|
|
|
|
|
|
|
|
|
|
public Queue<Disc> Queue = new Queue<Disc>();
|
|
|
|
|
|
|
|
|
|
public void Enqueue(Disc disc)
|
|
|
|
|
{
|
|
|
|
|
Queue.Enqueue(disc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Next()
|
|
|
|
|
{
|
|
|
|
|
if (Queue.Count != 0) Queue.Dequeue();
|
|
|
|
|
}
|
|
|
|
|
public void Eject()
|
|
|
|
|
{
|
|
|
|
|
CurrentDisc = null;
|
|
|
|
|
}
|
|
|
|
|
public void Insert()
|
|
|
|
|
{
|
|
|
|
|
if (Queue.Count > 0)
|
|
|
|
|
CurrentDisc = Queue.Peek();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Clear()
|
|
|
|
|
{
|
|
|
|
|
CurrentDisc = null;
|
|
|
|
|
Queue.Clear();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-14 21:56:54 +00:00
|
|
|
|
public partial class Disc
|
|
|
|
|
{
|
|
|
|
|
//main API to read a 2352-byte LBA from a disc.
|
|
|
|
|
//this starts at the beginning of the disc (at the lead-in)
|
|
|
|
|
//so add 150 to get to a FAD-address in the user data area
|
2011-06-02 01:52:10 +00:00
|
|
|
|
public void ReadLBA_2352(int lba, byte[] buffer, int offset)
|
2011-05-14 21:56:54 +00:00
|
|
|
|
{
|
|
|
|
|
if (lba < 150)
|
|
|
|
|
{
|
|
|
|
|
//lead-in area not supported yet
|
|
|
|
|
//in the future it will return something to mate with the
|
|
|
|
|
//subchannel data which we will load or calculate from the TOC
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Sectors[lba - 150].Sector.Read(buffer, offset);
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-02 01:52:10 +00:00
|
|
|
|
//main API to read a 2048-byte LBA from a disc.
|
|
|
|
|
//this starts at the beginning of the disc (at the lead-in)
|
|
|
|
|
//so add 150 to get to a FAD-address in the user data area
|
|
|
|
|
public void ReadLBA_2048(int lba, byte[] buffer, int offset)
|
|
|
|
|
{
|
|
|
|
|
if (lba < 150)
|
|
|
|
|
{
|
|
|
|
|
//lead-in area not supported yet
|
|
|
|
|
//in the future it will return something to mate with the
|
|
|
|
|
//subchannel data which we will load or calculate from the TOC
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
byte[] temp = new byte[2352];
|
|
|
|
|
Sectors[lba - 150].Sector.Read(temp, offset);
|
|
|
|
|
Array.Copy(temp, 16, buffer, offset, 2048);
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-14 21:56:54 +00:00
|
|
|
|
//main API to determine how many LBA sectors are available
|
|
|
|
|
public int LBACount { get { return Sectors.Count + 150; } }
|
|
|
|
|
|
|
|
|
|
//main api for reading the TOC from a disc
|
|
|
|
|
public DiscTOC ReadTOC()
|
|
|
|
|
{
|
|
|
|
|
return TOC;
|
|
|
|
|
}
|
2011-08-01 23:18:22 +00:00
|
|
|
|
|
2011-08-03 00:57:01 +00:00
|
|
|
|
// converts LBA to minute:second:frame format.
|
2011-08-01 23:18:22 +00:00
|
|
|
|
public static void ConvertLBAtoMSF(int lba, out byte m, out byte s, out byte f)
|
|
|
|
|
{
|
|
|
|
|
m = (byte) (lba / 75 / 60);
|
|
|
|
|
s = (byte) ((lba - (m * 75 * 60)) / 75);
|
|
|
|
|
f = (byte) (lba - (m * 75 * 60) - (s * 75));
|
|
|
|
|
}
|
2011-08-03 00:57:01 +00:00
|
|
|
|
|
|
|
|
|
// gets an identifying hash. hashes the first 512 sectors of
|
|
|
|
|
// the first data track on the disc.
|
|
|
|
|
public string GetHash()
|
|
|
|
|
{
|
|
|
|
|
byte[] buffer = new byte[512*2353];
|
|
|
|
|
foreach (var track in TOC.Sessions[0].Tracks)
|
|
|
|
|
{
|
|
|
|
|
if (track.TrackType == ETrackType.Audio)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
int lba_len = Math.Min(track.length_lba, 512);
|
|
|
|
|
for (int s=0; s<512 && s<track.length_lba; s++)
|
|
|
|
|
ReadLBA_2352(track.Indexes[1].lba + s, buffer, s*2352);
|
|
|
|
|
|
|
|
|
|
return Util.Hash_MD5(buffer, 0, lba_len*2352);
|
|
|
|
|
}
|
|
|
|
|
return "no data track found";
|
|
|
|
|
}
|
2011-05-14 21:56:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|