diff --git a/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_Compile.cs b/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_Compile.cs
index cc49815638..df03f3bc64 100644
--- a/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_Compile.cs
+++ b/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_Compile.cs
@@ -412,64 +412,54 @@ namespace BizHawk.Emulation.DiscSystem.CUE
//global cd text will acquire the cdtext commands set before track commands
curr_cdtext = OUT_GlobalCDText;
- for (int i = 0; i < cue.Commands.Count; i++)
+ foreach (var cmd in cue.Commands) switch (cmd)
{
- var cmd = cue.Commands[i];
-
- //these commands get dealt with globally. nothing to be done here
- //(but in the future we need to accumulate them into the compile pass output)
- if (cmd is CUE_File.Command.CATALOG || cmd is CUE_File.Command.CDTEXTFILE) continue;
-
- //nothing to be done for comments
- if (cmd is CUE_File.Command.REM) continue;
- if (cmd is CUE_File.Command.COMMENT) continue;
-
- //CD-text and related
- if (cmd is CUE_File.Command.PERFORMER performerCmd) curr_cdtext.Performer = performerCmd.Value;
- if (cmd is CUE_File.Command.SONGWRITER songwriterCmd) curr_cdtext.Songwriter = songwriterCmd.Value;
- if (cmd is CUE_File.Command.TITLE titleCmd) curr_cdtext.Title = titleCmd.Value;
- if (cmd is CUE_File.Command.ISRC isrcCmd) curr_cdtext.ISRC = isrcCmd.Value;
-
- //flags can only be set when a track command is running
- if (cmd is CUE_File.Command.FLAGS flagsCmd)
- {
- if (curr_track == null)
- Warn("Ignoring invalid flag commands outside of a track command");
- else
- //take care to |= it here, so the data flag doesn't get cleared
- curr_track.Flags |= flagsCmd.Flags;
- }
-
- if (cmd is CUE_File.Command.TRACK trackCmd)
- {
+ case CUE_File.Command.CATALOG:
+ case CUE_File.Command.CDTEXTFILE:
+ // these commands get dealt with globally. nothing to be done here
+ // (but in the future we need to accumulate them into the compile pass output)
+ continue;
+ case CUE_File.Command.REM:
+ case CUE_File.Command.COMMENT:
+ // nothing to be done for comments
+ continue;
+ case CUE_File.Command.PERFORMER performerCmd:
+ curr_cdtext.Performer = performerCmd.Value;
+ break;
+ case CUE_File.Command.SONGWRITER songwriterCmd:
+ curr_cdtext.Songwriter = songwriterCmd.Value;
+ break;
+ case CUE_File.Command.TITLE titleCmd:
+ curr_cdtext.Title = titleCmd.Value;
+ break;
+ case CUE_File.Command.ISRC isrcCmd:
+ curr_cdtext.ISRC = isrcCmd.Value;
+ break;
+ case CUE_File.Command.FLAGS flagsCmd:
+ // flags can only be set when a track command is running
+ if (curr_track == null) Warn("Ignoring invalid flag commands outside of a track command");
+ else curr_track.Flags |= flagsCmd.Flags; // take care to |= it here, so the data flag doesn't get cleared
+ break;
+ case CUE_File.Command.TRACK trackCmd:
CloseTrack();
OpenTrack(trackCmd);
- }
-
- if (cmd is CUE_File.Command.FILE fileCmd)
- {
+ break;
+ case CUE_File.Command.FILE fileCmd:
CloseFile();
OpenFile(fileCmd);
- }
-
- if (cmd is CUE_File.Command.INDEX indexCmd)
- {
- //todo - validate no postgap specified
+ break;
+ case CUE_File.Command.INDEX indexCmd:
+ //TODO validate no postgap specified
AddIndex(indexCmd);
- }
-
- if (cmd is CUE_File.Command.PREGAP pregapCmd)
- {
- //validate track open
- //validate no indexes
+ break;
+ case CUE_File.Command.PREGAP pregapCmd:
+ //TODO validate track open
+ //TODO validate no indexes
curr_track.PregapLength = pregapCmd.Length;
- }
-
- if (cmd is CUE_File.Command.POSTGAP postgapCmd)
- {
+ break;
+ case CUE_File.Command.POSTGAP postgapCmd:
curr_track.PostgapLength = postgapCmd.Length;
- }
-
+ break;
}
//it's a bit odd to close the file before closing the track, but...
diff --git a/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_File.cs b/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_File.cs
index 8d0154ff56..bfe25af663 100644
--- a/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_File.cs
+++ b/src/BizHawk.Emulation.DiscSystem/DiscFormats/CUE/CUE_File.cs
@@ -7,41 +7,171 @@ namespace BizHawk.Emulation.DiscSystem.CUE
///
internal class CUE_File
{
- // (here are all the commands we can encounter)
- public static class Command
+ ///
+ /// (here are all the commands we can encounter)
+ /// TODO record line number origin of command? Kind of nice but unessential
+ ///
+ public interface Command
{
- //TODO - record line number origin of command? Kind of nice but inessential
- public class CATALOG { public string Value; public override string ToString() { return $"CATALOG: {Value}"; } }
- public class CDTEXTFILE { public string Path; public override string ToString() { return $"CDTEXTFILE: {Path}"; } }
- public class FILE { public string Path; public CueFileType Type; public override string ToString() { return $"FILE ({Type}): {Path}"; } }
- public class FLAGS { public CueTrackFlags Flags; public override string ToString() { return $"FLAGS {Flags}"; } }
- public class INDEX { public int Number; public Timestamp Timestamp; public override string ToString() { return $"INDEX {Number,2} {Timestamp}"; } }
- public class ISRC { public string Value; public override string ToString() { return $"ISRC: {Value}"; } }
- public class PERFORMER { public string Value; public override string ToString() { return $"PERFORMER: {Value}"; } }
- public class POSTGAP { public Timestamp Length; public override string ToString() { return $"POSTGAP: {Length}"; } }
- public class PREGAP { public Timestamp Length; public override string ToString() { return $"PREGAP: {Length}"; } }
- public class REM { public string Value; public override string ToString() { return $"REM: {Value}"; } }
- public class COMMENT { public string Value; public override string ToString() { return $"COMMENT: {Value}"; } }
- public class SONGWRITER { public string Value; public override string ToString() { return $"SONGWRITER: {Value}"; } }
- public class TITLE { public string Value; public override string ToString() { return $"TITLE: {Value}"; } }
- public class TRACK { public int Number; public CueTrackType Type; public override string ToString() { return $"TRACK {Number,2} ({Type})"; } }
- }
+ public readonly struct CATALOG : Command
+ {
+ public readonly string Value;
+ public CATALOG(string value) => Value = value;
+
+ public override readonly string ToString() => $"CATALOG: {Value}";
+ }
+
+ public readonly struct CDTEXTFILE : Command
+ {
+ public readonly string Path;
+
+ public CDTEXTFILE(string path) => Path = path;
+
+ public override readonly string ToString() => $"CDTEXTFILE: {Path}";
+ }
+
+ public readonly struct FILE : Command
+ {
+ public readonly string Path;
+
+ public readonly CueFileType Type;
+
+ public FILE(string path, CueFileType type)
+ {
+ Path = path;
+ Type = type;
+ }
+
+ public override readonly string ToString() => $"FILE ({Type}): {Path}";
+ }
+
+ public readonly struct FLAGS : Command
+ {
+ public readonly CueTrackFlags Flags;
+
+ public FLAGS(CueTrackFlags flags) => Flags = flags;
+
+ public override readonly string ToString() => $"FLAGS {Flags}";
+ }
+
+ public readonly struct INDEX : Command
+ {
+ public readonly int Number;
+
+ public readonly Timestamp Timestamp;
+
+ public INDEX(int number, Timestamp timestamp)
+ {
+ Number = number;
+ Timestamp = timestamp;
+ }
+
+ public override readonly string ToString() => $"INDEX {Number,2} {Timestamp}";
+ }
+
+ public readonly struct ISRC : Command
+ {
+ public readonly string Value;
+
+ public ISRC(string value) => Value = value;
+
+ public override readonly string ToString() => $"ISRC: {Value}";
+ }
+
+ public readonly struct PERFORMER : Command
+ {
+ public readonly string Value;
+
+ public PERFORMER(string value) => Value = value;
+
+ public override readonly string ToString() => $"PERFORMER: {Value}";
+ }
+
+ public readonly struct POSTGAP : Command
+ {
+ public readonly Timestamp Length;
+
+ public POSTGAP(Timestamp length) => Length = length;
+
+ public override readonly string ToString() => $"POSTGAP: {Length}";
+ }
+
+ public readonly struct PREGAP : Command
+ {
+ public readonly Timestamp Length;
+
+ public PREGAP(Timestamp length) => Length = length;
+
+ public override readonly string ToString() => $"PREGAP: {Length}";
+ }
+
+ public readonly struct REM : Command
+ {
+ public readonly string Value;
+
+ public REM(string value) => Value = value;
+
+ public override readonly string ToString() => $"REM: {Value}";
+ }
+
+ public readonly struct COMMENT : Command
+ {
+ public readonly string Value;
+
+ public COMMENT(string value) => Value = value;
+
+ public override readonly string ToString() => $"COMMENT: {Value}";
+ }
+
+ public readonly struct SONGWRITER : Command
+ {
+ public readonly string Value;
+
+ public SONGWRITER(string value) => Value = value;
+
+ public override readonly string ToString() => $"SONGWRITER: {Value}";
+ }
+
+ public readonly struct TITLE : Command
+ {
+ public readonly string Value;
+
+ public TITLE(string value) => Value = value;
+
+ public override readonly string ToString() => $"TITLE: {Value}";
+ }
+
+ public readonly struct TRACK : Command
+ {
+ public readonly int Number;
+
+ public readonly CueTrackType Type;
+
+ public TRACK(int number, CueTrackType type)
+ {
+ Number = number;
+ Type = type;
+ }
+
+ public override readonly string ToString() => $"TRACK {Number,2} ({Type})";
+ }
+ }
///
/// Stuff other than the commands, global for the whole disc
///
public class DiscInfo
{
- public Command.CATALOG Catalog;
- public Command.ISRC ISRC;
- public Command.CDTEXTFILE CDTextFile;
+ public Command.CATALOG? Catalog;
+ public Command.ISRC? ISRC;
+ public Command.CDTEXTFILE? CDTextFile;
}
///
/// The sequential list of commands parsed out of the cue file
///
- public List