631 lines
21 KiB
C#
631 lines
21 KiB
C#
using System;
|
|
using System.Text;
|
|
using System.IO;
|
|
using System.Globalization;
|
|
using System.Collections.Generic;
|
|
|
|
//check out ccd2iso linux program?
|
|
//https://wiki.archlinux.org/index.php/CD_Burning#TOC.2FCUE.2FBIN_for_mixed-mode_disks advice here
|
|
//also referencing mednafen sources
|
|
|
|
//TODO - copy mednafen sanity checking
|
|
//TODO - check subcode vs TOC (warnings if different)
|
|
//TODO - check TOC vs tracks (warnings if different)
|
|
|
|
namespace BizHawk.Emulation.DiscSystem
|
|
{
|
|
public class CCD_Format
|
|
{
|
|
/// <summary>
|
|
/// Represents a CCD file, faithfully. Minimal interpretation of the data happens.
|
|
/// Currently the [TRACK] sections aren't parsed, though.
|
|
/// </summary>
|
|
public class CCDFile
|
|
{
|
|
/// <summary>
|
|
/// which version CCD file this came from. We hope it shouldn't affect the semantics of anything else in here, but just in case..
|
|
/// </summary>
|
|
public int Version;
|
|
|
|
/// <summary>
|
|
/// this is probably a 0 or 1 bool
|
|
/// </summary>
|
|
public int DataTracksScrambled;
|
|
|
|
/// <summary>
|
|
/// ???
|
|
/// </summary>
|
|
public int CDTextLength;
|
|
|
|
/// <summary>
|
|
/// The [Session] sections
|
|
/// </summary>
|
|
public List<CCDSession> Sessions = new List<CCDSession>();
|
|
|
|
/// <summary>
|
|
/// The [Entry] sctions
|
|
/// </summary>
|
|
public List<CCDTocEntry> TOCEntries = new List<CCDTocEntry>();
|
|
|
|
/// <summary>
|
|
/// The [TRACK] sections
|
|
/// </summary>
|
|
public List<CCDTrack> Tracks = new List<CCDTrack>();
|
|
|
|
/// <summary>
|
|
/// The [TRACK] sections, indexed by number
|
|
/// </summary>
|
|
public Dictionary<int, CCDTrack> TracksByNumber = new Dictionary<int, CCDTrack>();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Represents an [Entry] section from a CCD file
|
|
/// </summary>
|
|
public class CCDTocEntry
|
|
{
|
|
public CCDTocEntry(int entryNum)
|
|
{
|
|
EntryNum = entryNum;
|
|
}
|
|
|
|
/// <summary>
|
|
/// these should be 0-indexed
|
|
/// </summary>
|
|
public int EntryNum;
|
|
|
|
|
|
/// <summary>
|
|
/// the CCD specifies this, but it isnt in the actual disc data as such, it is encoded some other (likely difficult to extract) way and thats why CCD puts it here
|
|
/// </summary>
|
|
public int Session;
|
|
|
|
/// <summary>
|
|
/// this seems just to be the LBA corresponding to AMIN:ASEC:AFRAME (give or take 150). It's not stored on the disc, and it's redundant.
|
|
/// </summary>
|
|
public int ALBA;
|
|
|
|
/// <summary>
|
|
/// this seems just to be the LBA corresponding to PMIN:PSEC:PFRAME (give or take 150). It's not stored on the disc, and it's redundant.
|
|
/// </summary>
|
|
public int PLBA;
|
|
|
|
//these correspond pretty directly to values in the Q subchannel fields
|
|
//NOTE: they're specified as absolute MSF. That means, they're 2 seconds off from what they should be when viewed as final TOC values
|
|
public int Control;
|
|
public int ADR;
|
|
public int TrackNo;
|
|
public int Point;
|
|
public int AMin;
|
|
public int ASec;
|
|
public int AFrame;
|
|
public int Zero;
|
|
public int PMin;
|
|
public int PSec;
|
|
public int PFrame;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Represents a [Track] section from a CCD file
|
|
/// </summary>
|
|
public class CCDTrack
|
|
{
|
|
public CCDTrack(int number)
|
|
{
|
|
this.Number = number;
|
|
}
|
|
|
|
/// <summary>
|
|
/// note: this is 1-indexed
|
|
/// </summary>
|
|
public int Number;
|
|
|
|
/// <summary>
|
|
/// The specified data mode.
|
|
/// </summary>
|
|
public int Mode;
|
|
|
|
/// <summary>
|
|
/// The indexes specified for the track (these are 0-indexed)
|
|
/// </summary>
|
|
public Dictionary<int, int> Indexes = new Dictionary<int, int>();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Represents a [Session] section from a CCD file
|
|
/// </summary>
|
|
public class CCDSession
|
|
{
|
|
public CCDSession(int number)
|
|
{
|
|
this.Number = number;
|
|
}
|
|
|
|
/// <summary>
|
|
/// note: this is 1-indexed.
|
|
/// </summary>
|
|
public int Number;
|
|
|
|
//Not sure what the default should be.. ive only seen mode=2
|
|
public int PregapMode;
|
|
|
|
/// <summary>
|
|
/// this is probably a 0 or 1 bool
|
|
/// </summary>
|
|
public int PregapSubcode;
|
|
}
|
|
|
|
public class CCDParseException : Exception
|
|
{
|
|
public CCDParseException(string message) : base(message) { }
|
|
}
|
|
|
|
class CCDSection : Dictionary<string,int>
|
|
{
|
|
public string Name;
|
|
public int FetchOrDefault(int def, string key)
|
|
{
|
|
TryGetValue(key, out def);
|
|
return def;
|
|
}
|
|
public int FetchOrFail(string key)
|
|
{
|
|
int ret;
|
|
if(!TryGetValue(key, out ret))
|
|
throw new CCDParseException($"Malformed or unexpected CCD format: missing required [Entry] key: {key}");
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
List<CCDSection> ParseSections(Stream stream)
|
|
{
|
|
List<CCDSection> sections = new List<CCDSection>();
|
|
|
|
//TODO - do we need to attempt to parse out the version tag in a first pass?
|
|
//im doing this from a version 3 example
|
|
|
|
StreamReader sr = new StreamReader(stream);
|
|
CCDSection currSection = null;
|
|
for (; ; )
|
|
{
|
|
var line = sr.ReadLine();
|
|
if (line == null) break;
|
|
if (line == "") continue;
|
|
if (line.StartsWith("["))
|
|
{
|
|
currSection = new CCDSection();
|
|
currSection.Name = line.Trim('[', ']').ToUpper();
|
|
sections.Add(currSection);
|
|
}
|
|
else
|
|
{
|
|
var parts = line.Split('=');
|
|
if (parts.Length != 2)
|
|
throw new CCDParseException("Malformed or unexpected CCD format: parsing item into two parts");
|
|
int val;
|
|
if (parts[1].StartsWith("0x") || parts[1].StartsWith("0X"))
|
|
val = int.Parse(parts[1].Substring(2), NumberStyles.HexNumber);
|
|
else val = int.Parse(parts[1]);
|
|
currSection[parts[0].ToUpper()] = val;
|
|
}
|
|
} //loop until lines exhausted
|
|
|
|
return sections;
|
|
}
|
|
|
|
int PreParseIntegrityCheck(List<CCDSection> sections)
|
|
{
|
|
if (sections.Count == 0) throw new CCDParseException("Malformed CCD format: no sections");
|
|
|
|
//we need at least a CloneCD and Disc section
|
|
if (sections.Count < 2) throw new CCDParseException("Malformed CCD format: insufficient sections");
|
|
|
|
var ccdSection = sections[0];
|
|
if (ccdSection.Name != "CLONECD")
|
|
throw new CCDParseException("Malformed CCD format: confusing first section name");
|
|
|
|
if (!ccdSection.ContainsKey("VERSION"))
|
|
throw new CCDParseException("Malformed CCD format: missing version in CloneCD section");
|
|
|
|
if(sections[1].Name != "DISC")
|
|
throw new CCDParseException("Malformed CCD format: section[1] isn't [Disc]");
|
|
|
|
int version = ccdSection["VERSION"];
|
|
|
|
return version;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parses a CCD file contained in the provided stream
|
|
/// </summary>
|
|
public CCDFile ParseFrom(Stream stream)
|
|
{
|
|
CCDFile ccdf = new CCDFile();
|
|
|
|
var sections = ParseSections(stream);
|
|
ccdf.Version = PreParseIntegrityCheck(sections);
|
|
|
|
var discSection = sections[1];
|
|
int nTocEntries = discSection["TOCENTRIES"]; //its conceivable that this could be missing
|
|
int nSessions = discSection["SESSIONS"]; //its conceivable that this could be missing
|
|
ccdf.DataTracksScrambled = discSection.FetchOrDefault(0, "DATATRACKSSCRAMBLED");
|
|
ccdf.CDTextLength = discSection.FetchOrDefault(0, "CDTEXTLENGTH");
|
|
|
|
if (ccdf.DataTracksScrambled==1) throw new CCDParseException($"Malformed CCD format: {nameof(ccdf.DataTracksScrambled)}=1 not supported. Please report this, so we can understand what it means.");
|
|
|
|
for (int i = 2; i < sections.Count; i++)
|
|
{
|
|
var section = sections[i];
|
|
if (section.Name.StartsWith("SESSION"))
|
|
{
|
|
int sesnum = int.Parse(section.Name.Split(' ')[1]);
|
|
CCDSession session = new CCDSession(sesnum);
|
|
ccdf.Sessions.Add(session);
|
|
if (sesnum != ccdf.Sessions.Count)
|
|
throw new CCDParseException("Malformed CCD format: wrong session number in sequence");
|
|
session.PregapMode = section.FetchOrDefault(0, "PREGAPMODE");
|
|
session.PregapSubcode = section.FetchOrDefault(0, "PREGAPSUBC");
|
|
}
|
|
else if (section.Name.StartsWith("ENTRY"))
|
|
{
|
|
int entryNum = int.Parse(section.Name.Split(' ')[1]);
|
|
CCDTocEntry entry = new CCDTocEntry(entryNum);
|
|
ccdf.TOCEntries.Add(entry);
|
|
|
|
entry.Session = section.FetchOrFail("SESSION");
|
|
entry.Point = section.FetchOrFail("POINT");
|
|
entry.ADR = section.FetchOrFail("ADR");
|
|
entry.Control = section.FetchOrFail("CONTROL");
|
|
entry.TrackNo = section.FetchOrFail("TRACKNO");
|
|
entry.AMin = section.FetchOrFail("AMIN");
|
|
entry.ASec = section.FetchOrFail("ASEC");
|
|
entry.AFrame = section.FetchOrFail("AFRAME");
|
|
entry.ALBA = section.FetchOrFail("ALBA");
|
|
entry.Zero = section.FetchOrFail("ZERO");
|
|
entry.PMin = section.FetchOrFail("PMIN");
|
|
entry.PSec = section.FetchOrFail("PSEC");
|
|
entry.PFrame = section.FetchOrFail("PFRAME");
|
|
entry.PLBA = section.FetchOrFail("PLBA");
|
|
|
|
//note: LBA 0 is Ansolute MSF 00:02:00
|
|
if (new Timestamp(entry.AMin, entry.ASec, entry.AFrame).Sector != entry.ALBA + 150)
|
|
throw new CCDParseException("Warning: inconsistency in CCD ALBA vs computed A MSF");
|
|
if (new Timestamp(entry.PMin, entry.PSec, entry.PFrame).Sector != entry.PLBA + 150)
|
|
throw new CCDParseException("Warning: inconsistency in CCD PLBA vs computed P MSF");
|
|
|
|
if(entry.Session != 1)
|
|
throw new CCDParseException("Malformed CCD format: not yet supporting multi-session files");
|
|
}
|
|
else if (section.Name.StartsWith("TRACK"))
|
|
{
|
|
int entryNum = int.Parse(section.Name.Split(' ')[1]);
|
|
CCDTrack track = new CCDTrack(entryNum);
|
|
ccdf.Tracks.Add(track);
|
|
ccdf.TracksByNumber[entryNum] = track;
|
|
foreach (var kvp in section)
|
|
{
|
|
if (kvp.Key == "MODE")
|
|
track.Mode = kvp.Value;
|
|
if (kvp.Key.StartsWith("INDEX"))
|
|
{
|
|
int inum = int.Parse(kvp.Key.Split(' ')[1]);
|
|
track.Indexes[inum] = kvp.Value;
|
|
}
|
|
}
|
|
}
|
|
} //sections loop
|
|
|
|
return ccdf;
|
|
}
|
|
|
|
public class LoadResults
|
|
{
|
|
public List<RawTOCEntry> RawTOCEntries;
|
|
public CCDFile ParsedCCDFile;
|
|
public bool Valid;
|
|
public Exception FailureException;
|
|
public string ImgPath;
|
|
public string SubPath;
|
|
public string CcdPath;
|
|
}
|
|
|
|
public static LoadResults LoadCCDPath(string path)
|
|
{
|
|
LoadResults ret = new LoadResults();
|
|
ret.CcdPath = path;
|
|
ret.ImgPath = Path.ChangeExtension(path, ".img");
|
|
ret.SubPath = Path.ChangeExtension(path, ".sub");
|
|
try
|
|
{
|
|
if (!File.Exists(path)) throw new CCDParseException("Malformed CCD format: nonexistent CCD file!");
|
|
|
|
CCDFile ccdf;
|
|
using (var infCCD = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read))
|
|
ccdf = new CCD_Format().ParseFrom(infCCD);
|
|
|
|
ret.ParsedCCDFile = ccdf;
|
|
|
|
ret.Valid = true;
|
|
}
|
|
catch (CCDParseException ex)
|
|
{
|
|
ret.FailureException = ex;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
public static void Dump(Disc disc, string path)
|
|
{
|
|
using (var sw = new StreamWriter(path))
|
|
{
|
|
//NOTE: IsoBuster requires the A0,A1,A2 RawTocEntries to be first or else it can't do anything with the tracks
|
|
//if we ever get them in a different order, we'll have to re-order them here
|
|
|
|
sw.WriteLine("[CloneCD]");
|
|
sw.WriteLine("Version=3");
|
|
sw.WriteLine();
|
|
sw.WriteLine("[Disc]");
|
|
sw.WriteLine("TocEntries={0}", disc.RawTOCEntries.Count);
|
|
sw.WriteLine("Sessions=1");
|
|
sw.WriteLine("DataTracksScrambled=0");
|
|
sw.WriteLine("CDTextLength=0"); //not supported anyway
|
|
sw.WriteLine();
|
|
sw.WriteLine("[Session 1]");
|
|
sw.WriteLine("PreGapMode=2");
|
|
sw.WriteLine("PreGapSubC=1");
|
|
sw.WriteLine();
|
|
for (int i = 0; i < disc.RawTOCEntries.Count; i++)
|
|
{
|
|
var entry = disc.RawTOCEntries[i];
|
|
|
|
//ehhh something's wrong with how I track these
|
|
int point = entry.QData.q_index.DecimalValue;
|
|
if (point == 100) point = 0xA0;
|
|
if (point == 101) point = 0xA1;
|
|
if (point == 102) point = 0xA2;
|
|
|
|
sw.WriteLine("[Entry {0}]", i);
|
|
sw.WriteLine("Session=1");
|
|
sw.WriteLine("Point=0x{0:x2}", point);
|
|
sw.WriteLine("ADR=0x{0:x2}", entry.QData.ADR);
|
|
sw.WriteLine("Control=0x{0:x2}", (int)entry.QData.CONTROL);
|
|
sw.WriteLine("TrackNo={0}", entry.QData.q_tno.DecimalValue);
|
|
sw.WriteLine("AMin={0}", entry.QData.min.DecimalValue);
|
|
sw.WriteLine("ASec={0}", entry.QData.sec.DecimalValue);
|
|
sw.WriteLine("AFrame={0}", entry.QData.frame.DecimalValue);
|
|
sw.WriteLine("ALBA={0}", entry.QData.Timestamp - 150); //remember to adapt the absolute MSF to an LBA (this field is redundant...)
|
|
sw.WriteLine("Zero={0}", entry.QData.zero);
|
|
sw.WriteLine("PMin={0}", entry.QData.ap_min.DecimalValue);
|
|
sw.WriteLine("PSec={0}", entry.QData.ap_sec.DecimalValue);
|
|
sw.WriteLine("PFrame={0}", entry.QData.ap_frame.DecimalValue);
|
|
sw.WriteLine("PLBA={0}", entry.QData.AP_Timestamp - 150); //remember to adapt the absolute MSF to an LBA (this field is redundant...)
|
|
sw.WriteLine();
|
|
}
|
|
|
|
//this is nonsense, really. the whole CCD track list shouldn't be needed.
|
|
//but in order to make a high quality CCD which can be inspected by various other tools, we need it
|
|
//now, regarding the indexes.. theyre truly useless. having indexes written out with the tracks is bad news.
|
|
//index information is only truly stored in subQ
|
|
for (int tnum = 1; tnum <= disc.Session1.LastInformationTrack.Number; tnum++)
|
|
{
|
|
var track = disc.Session1.Tracks[tnum];
|
|
sw.WriteLine("[TRACK {0}]", track.Number);
|
|
sw.WriteLine("MODE={0}", track.Mode);
|
|
//indexes are BS, dont write them. but we certainly need an index 1
|
|
sw.WriteLine("INDEX 1={0}", track.LBA);
|
|
sw.WriteLine();
|
|
}
|
|
}
|
|
|
|
//TODO - actually re-add
|
|
//dump the img and sub
|
|
//TODO - acquire disk size first
|
|
string imgPath = Path.ChangeExtension(path, ".img");
|
|
string subPath = Path.ChangeExtension(path, ".sub");
|
|
var buf2448 = new byte[2448];
|
|
DiscSectorReader dsr = new DiscSectorReader(disc);
|
|
|
|
using (var imgFile = File.OpenWrite(imgPath))
|
|
using (var subFile = File.OpenWrite(subPath))
|
|
{
|
|
|
|
int nLBA = disc.Session1.LeadoutLBA;
|
|
for (int lba = 0; lba < nLBA; lba++)
|
|
{
|
|
dsr.ReadLBA_2448(lba, buf2448, 0);
|
|
imgFile.Write(buf2448, 0, 2352);
|
|
subFile.Write(buf2448, 2352, 96);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
class SS_CCD : ISectorSynthJob2448
|
|
{
|
|
public void Synth(SectorSynthJob job)
|
|
{
|
|
//CCD is always containing everything we'd need (unless a .sub is missing?) so don't worry about flags
|
|
var imgBlob = job.Disc.DisposableResources[0] as IBlob;
|
|
var subBlob = job.Disc.DisposableResources[1] as IBlob;
|
|
//Read_2442(job.LBA, job.DestBuffer2448, job.DestOffset);
|
|
|
|
//read the IMG data if needed
|
|
if ((job.Parts & ESectorSynthPart.UserAny) != 0)
|
|
{
|
|
long ofs = job.LBA * 2352;
|
|
imgBlob.Read(ofs, job.DestBuffer2448, 0, 2352);
|
|
}
|
|
|
|
//if subcode is needed, read it
|
|
if ((job.Parts & (ESectorSynthPart.SubcodeAny)) != 0)
|
|
{
|
|
long ofs = job.LBA * 96;
|
|
subBlob.Read(ofs, job.DestBuffer2448, 2352, 96);
|
|
|
|
//subcode comes to us deinterleved; we may still need to interleave it
|
|
if ((job.Parts & (ESectorSynthPart.SubcodeDeinterleave)) == 0)
|
|
{
|
|
SynthUtils.InterleaveSubcodeInplace(job.DestBuffer2448, job.DestOffset + 2352);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Loads a CCD at the specified path to a Disc object
|
|
/// </summary>
|
|
public Disc LoadCCDToDisc(string ccdPath, DiscMountPolicy IN_DiscMountPolicy)
|
|
{
|
|
var loadResults = LoadCCDPath(ccdPath);
|
|
if (!loadResults.Valid)
|
|
throw loadResults.FailureException;
|
|
|
|
Disc disc = new Disc();
|
|
|
|
IBlob imgBlob = null, subBlob = null;
|
|
long imgLen = -1, subLen;
|
|
|
|
//mount the IMG file
|
|
//first check for a .ecm in place of the img
|
|
var imgPath = loadResults.ImgPath;
|
|
if (!File.Exists(imgPath))
|
|
{
|
|
var ecmPath = Path.ChangeExtension(imgPath, ".img.ecm");
|
|
if (File.Exists(ecmPath))
|
|
{
|
|
if (Disc.Blob_ECM.IsECM(ecmPath))
|
|
{
|
|
var ecm = new Disc.Blob_ECM();
|
|
ecm.Load(ecmPath);
|
|
imgBlob = ecm;
|
|
imgLen = ecm.Length;
|
|
}
|
|
}
|
|
}
|
|
if (imgBlob == null)
|
|
{
|
|
if (!File.Exists(loadResults.ImgPath)) throw new CCDParseException("Malformed CCD format: nonexistent IMG file!");
|
|
var imgFile = new Disc.Blob_RawFile() { PhysicalPath = loadResults.ImgPath };
|
|
imgLen = imgFile.Length;
|
|
imgBlob = imgFile;
|
|
}
|
|
disc.DisposableResources.Add(imgBlob);
|
|
|
|
//mount the SUB file
|
|
if (!File.Exists(loadResults.SubPath)) throw new CCDParseException("Malformed CCD format: nonexistent SUB file!");
|
|
var subFile = new Disc.Blob_RawFile() { PhysicalPath = loadResults.SubPath };
|
|
subBlob = subFile;
|
|
disc.DisposableResources.Add(subBlob);
|
|
subLen = subFile.Length;
|
|
|
|
//quick integrity check of file sizes
|
|
if (imgLen % 2352 != 0) throw new CCDParseException("Malformed CCD format: IMG file length not multiple of 2352");
|
|
int NumImgSectors = (int)(imgLen / 2352);
|
|
if (subLen != NumImgSectors * 96) throw new CCDParseException("Malformed CCD format: SUB file length not matching IMG");
|
|
|
|
var ccdf = loadResults.ParsedCCDFile;
|
|
|
|
//the only instance of a sector synthesizer we'll need
|
|
SS_CCD synth = new SS_CCD();
|
|
|
|
//generate DiscTOCRaw items from the ones specified in the CCD file
|
|
//TODO - range validate these (too many truncations to byte)
|
|
disc.RawTOCEntries = new List<RawTOCEntry>();
|
|
foreach (var entry in ccdf.TOCEntries)
|
|
{
|
|
BCD2 tno, ino;
|
|
|
|
//this should actually be zero. im not sure if this is stored as BCD2 or not
|
|
tno = BCD2.FromDecimal(entry.TrackNo);
|
|
|
|
//these are special values.. I think, taken from this:
|
|
//http://www.staff.uni-mainz.de/tacke/scsi/SCSI2-14.html
|
|
//the CCD will contain Points as decimal values except for these specially converted decimal values which should stay as BCD.
|
|
//Why couldn't they all be BCD? I don't know. I guess because BCD is inconvenient, but only A0 and friends have special meaning. It's confusing.
|
|
ino = BCD2.FromDecimal(entry.Point);
|
|
if (entry.Point == 0xA0) ino.BCDValue = 0xA0;
|
|
else if (entry.Point == 0xA1) ino.BCDValue = 0xA1;
|
|
else if (entry.Point == 0xA2) ino.BCDValue = 0xA2;
|
|
|
|
var q = new SubchannelQ
|
|
{
|
|
q_status = SubchannelQ.ComputeStatus(entry.ADR, (EControlQ)(entry.Control & 0xF)),
|
|
q_tno = tno,
|
|
q_index = ino,
|
|
min = BCD2.FromDecimal(entry.AMin),
|
|
sec = BCD2.FromDecimal(entry.ASec),
|
|
frame = BCD2.FromDecimal(entry.AFrame),
|
|
zero = (byte)entry.Zero,
|
|
ap_min = BCD2.FromDecimal(entry.PMin),
|
|
ap_sec = BCD2.FromDecimal(entry.PSec),
|
|
ap_frame = BCD2.FromDecimal(entry.PFrame),
|
|
q_crc = 0, //meaningless
|
|
};
|
|
|
|
disc.RawTOCEntries.Add(new RawTOCEntry { QData = q });
|
|
}
|
|
|
|
//analyze the RAWTocEntries to figure out what type of track track 1 is
|
|
var tocSynth = new Synthesize_DiscTOC_From_RawTOCEntries_Job() { Entries = disc.RawTOCEntries };
|
|
tocSynth.Run();
|
|
|
|
//Add sectors for the mandatory track 1 pregap, which isn't stored in the CCD file
|
|
//We reuse some CUE code for this.
|
|
//If we load other formats later we might should abstract this even further (to a synthesizer job)
|
|
//It can't really be abstracted from cue files though due to the necessity of merging this with other track 1 pregaps
|
|
CUE.CueTrackType pregapTrackType = CUE.CueTrackType.Audio;
|
|
if (tocSynth.Result.TOCItems[1].IsData)
|
|
{
|
|
if (tocSynth.Result.Session1Format == SessionFormat.Type20_CDXA)
|
|
pregapTrackType = CUE.CueTrackType.Mode2_2352;
|
|
else if (tocSynth.Result.Session1Format == SessionFormat.Type10_CDI)
|
|
pregapTrackType = CUE.CueTrackType.CDI_2352;
|
|
else if (tocSynth.Result.Session1Format == SessionFormat.Type00_CDROM_CDDA)
|
|
pregapTrackType = CUE.CueTrackType.Mode1_2352;
|
|
}
|
|
for (int i = 0; i < 150; i++)
|
|
{
|
|
var ss_gap = new CUE.SS_Gap()
|
|
{
|
|
Policy = IN_DiscMountPolicy,
|
|
TrackType = pregapTrackType
|
|
};
|
|
disc._Sectors.Add(ss_gap);
|
|
|
|
int qRelMSF = i - 150;
|
|
|
|
//tweak relMSF due to ambiguity/contradiction in yellowbook docs
|
|
if (!IN_DiscMountPolicy.CUE_PregapContradictionModeA)
|
|
qRelMSF++;
|
|
|
|
//setup subQ
|
|
byte ADR = 1; //absent some kind of policy for how to set it, this is a safe assumption:
|
|
ss_gap.sq.SetStatus(ADR, tocSynth.Result.TOCItems[1].Control);
|
|
ss_gap.sq.q_tno = BCD2.FromDecimal(1);
|
|
ss_gap.sq.q_index = BCD2.FromDecimal(0);
|
|
ss_gap.sq.AP_Timestamp = i;
|
|
ss_gap.sq.Timestamp = qRelMSF;
|
|
|
|
//setup subP
|
|
ss_gap.Pause = true;
|
|
}
|
|
|
|
//build the sectors:
|
|
//set up as many sectors as we have img/sub for, even if the TOC doesnt reference them
|
|
//(the TOC is unreliable, and the Track records are redundant)
|
|
for (int i = 0; i < NumImgSectors; i++)
|
|
{
|
|
disc._Sectors.Add(synth);
|
|
}
|
|
|
|
return disc;
|
|
}
|
|
|
|
|
|
|
|
} //class CCD_Format
|
|
}
|
|
|
|
|