Files
Aaru/Aaru.Images/DiscJuggler/Read.cs

1341 lines
52 KiB
C#
Raw Normal View History

// /***************************************************************************
2020-02-27 12:31:25 +00:00
// Aaru Data Preservation Suite
// ----------------------------------------------------------------------------
//
// Filename : Read.cs
// Author(s) : Natalia Portillo <claunia@claunia.com>
//
// Component : Disk image plugins.
//
// --[ Description ] ----------------------------------------------------------
//
// Reads DiscJuggler disc images.
//
// --[ License ] --------------------------------------------------------------
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation; either version 2.1 of the
// License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, see <http://www.gnu.org/licenses/>.
//
// ----------------------------------------------------------------------------
2024-12-19 10:45:18 +00:00
// Copyright © 2011-2025 Natalia Portillo
// ****************************************************************************/
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
2020-02-27 00:33:26 +00:00
using Aaru.CommonTypes;
using Aaru.CommonTypes.Enums;
using Aaru.CommonTypes.Interfaces;
using Aaru.CommonTypes.Structs;
using Aaru.Decoders.CD;
2020-07-20 15:43:52 +01:00
using Aaru.Helpers;
using Aaru.Logging;
2020-02-27 00:33:26 +00:00
using Session = Aaru.CommonTypes.Structs.Session;
namespace Aaru.Images;
2022-03-06 13:29:38 +00:00
public sealed partial class DiscJuggler
{
2023-10-03 23:34:59 +01:00
#region IOpticalMediaImage Members
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber Open(IFilter imageFilter)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
_imageStream = imageFilter.GetDataForkStream();
// Read size of image descriptor
_imageStream.Seek(-4, SeekOrigin.End);
byte[] dscLenB = new byte[4];
_imageStream.EnsureRead(dscLenB, 0, 4);
int dscLen = BitConverter.ToInt32(dscLenB, 0);
2022-03-06 13:29:38 +00:00
2024-05-01 04:05:22 +01:00
if(dscLen >= _imageStream.Length) return ErrorNumber.InvalidArgument;
2022-03-06 13:29:38 +00:00
byte[] descriptor = new byte[dscLen];
2022-03-06 13:29:38 +00:00
_imageStream.Seek(-dscLen, SeekOrigin.End);
_imageStream.EnsureRead(descriptor, 0, dscLen);
2022-03-06 13:29:38 +00:00
// Sessions
2024-05-01 04:05:22 +01:00
if(descriptor[0] > 99 || descriptor[0] == 0) return ErrorNumber.InvalidArgument;
2022-03-06 13:29:38 +00:00
int position = 1;
2022-03-06 13:29:38 +00:00
ushort sessionSequence = 0;
2024-05-01 04:39:38 +01:00
Sessions = [];
Tracks = [];
Partitions = [];
2022-03-06 13:29:38 +00:00
_offsetMap = new Dictionary<uint, ulong>();
_trackFlags = new Dictionary<uint, byte>();
ushort mediumType;
byte maxS = descriptor[0];
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "maxS = {0}", maxS);
2022-03-06 13:29:38 +00:00
uint lastSessionTrack = 0;
ulong currentOffset = 0;
// Read sessions
for(byte s = 0; s <= maxS; s++)
2017-12-19 20:33:03 +00:00
{
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "s = {0}", s);
2022-03-06 13:29:38 +00:00
// Seems all sessions start with this data
if(descriptor[position + 0] != 0x00 ||
descriptor[position + 2] != 0x00 ||
descriptor[position + 3] != 0x00 ||
descriptor[position + 4] != 0x00 ||
descriptor[position + 5] != 0x00 ||
descriptor[position + 6] != 0x00 ||
descriptor[position + 7] != 0x00 ||
descriptor[position + 8] != 0x00 ||
descriptor[position + 9] != 0x01 ||
descriptor[position + 10] != 0x00 ||
descriptor[position + 11] != 0x00 ||
descriptor[position + 12] != 0x00 ||
descriptor[position + 13] != 0xFF ||
descriptor[position + 14] != 0xFF)
2017-12-19 20:33:03 +00:00
{
bool nextFound = false;
2022-03-06 13:29:38 +00:00
// But on generated (not dumped) image, it can have some data between last written session and
// next open one, so depend on if we already got a track
while(position + 16 < descriptor.Length)
{
if(descriptor[position + 0] != 0x00 ||
descriptor[position + 2] != 0x00 ||
descriptor[position + 3] != 0x00 ||
descriptor[position + 4] != 0x00 ||
descriptor[position + 5] != 0x00 ||
descriptor[position + 6] != 0x00 ||
descriptor[position + 7] != 0x00 ||
descriptor[position + 8] != 0x00 ||
descriptor[position + 9] != 0x01 ||
descriptor[position + 10] != 0x00 ||
descriptor[position + 11] != 0x00 ||
descriptor[position + 12] != 0x00 ||
descriptor[position + 13] != 0xFF ||
descriptor[position + 14] != 0xFF)
{
2022-03-06 13:29:38 +00:00
position++;
2022-03-06 13:29:38 +00:00
continue;
}
2022-03-06 13:29:38 +00:00
nextFound = true;
break;
}
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(!nextFound) return Tracks.Count > 0 ? ErrorNumber.NoError : ErrorNumber.InvalidArgument;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
position += 15;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
2022-03-06 13:29:38 +00:00
// Too many tracks
2024-05-01 04:05:22 +01:00
if(descriptor[position + 1] > 99) return ErrorNumber.InvalidArgument;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
byte maxT = descriptor[position + 1];
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "maxT = {0}", maxT);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
sessionSequence++;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
var session = new Session
{
Sequence = sessionSequence,
EndTrack = uint.MinValue,
StartTrack = uint.MaxValue
};
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
position += 15;
bool addedATrack = false;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Read track
for(byte t = 0; t < maxT; t++)
{
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "t = {0}", t);
2022-03-06 13:29:38 +00:00
var track = new Track();
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 16;
2017-12-19 20:33:03 +00:00
byte[] trackFilenameB = new byte[descriptor[position]];
2022-03-06 13:29:38 +00:00
position++;
Array.Copy(descriptor, position, trackFilenameB, 0, trackFilenameB.Length);
position += trackFilenameB.Length;
track.File = Path.GetFileName(Encoding.Default.GetString(trackFilenameB));
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tfilename = {0}", track.File);
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 29;
mediumType = BitConverter.ToUInt16(descriptor, position);
position += 2;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tmediumType = {0}", mediumType);
2022-03-06 13:29:38 +00:00
// Read indices
ushort maxI = BitConverter.ToUInt16(descriptor, position);
2022-03-06 13:29:38 +00:00
position += 2;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tmaxI = {0}", maxI);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// This is not really the index position, but, the index length, go figure
for(ushort i = 0; i < maxI; i++)
{
int index = BitConverter.ToInt32(descriptor, position);
2022-03-06 13:29:38 +00:00
track.Indexes.Add(i, index);
2018-06-22 08:08:38 +01:00
position += 4;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tindex[{1}] = {0}", index, i);
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
// Read CD-Text
uint maxC = BitConverter.ToUInt32(descriptor, position);
2022-03-06 13:29:38 +00:00
position += 4;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tmaxC = {0}", maxC);
2022-03-06 13:29:38 +00:00
for(uint c = 0; c < maxC; c++)
{
for(int cb = 0; cb < 18; cb++)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
int bLen = descriptor[position];
position++;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tc[{1}][{2}].Length = {0}", bLen, c, cb);
2024-05-01 04:05:22 +01:00
if(bLen <= 0) continue;
byte[] textBlk = new byte[bLen];
2022-03-06 13:29:38 +00:00
Array.Copy(descriptor, position, textBlk, 0, bLen);
position += bLen;
2022-03-06 13:29:38 +00:00
// Track title
2024-05-01 04:05:22 +01:00
if(cb != 10) continue;
2022-03-06 13:29:38 +00:00
track.Description = Encoding.Default.GetString(textBlk, 0, bLen);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\t" + Localization.Track_title_0, track.Description);
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
position += 2;
uint trackMode = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\ttrackMode = {0}", trackMode);
2022-03-06 13:29:38 +00:00
position += 4;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 4;
2022-03-06 13:29:38 +00:00
session.Sequence = (ushort)(BitConverter.ToUInt32(descriptor, position) + 1);
track.Session = session.Sequence;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tsession = {0}", session.Sequence);
2022-03-06 13:29:38 +00:00
position += 4;
track.Sequence = BitConverter.ToUInt32(descriptor, position) + lastSessionTrack + 1;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME,
2024-05-01 04:05:22 +01:00
"\ttrack = {1} + {2} + 1 = {0}",
track.Sequence,
BitConverter.ToUInt32(descriptor, position),
lastSessionTrack);
2022-03-06 13:29:38 +00:00
// There's always an index 0 in the image
track.Pregap = (ulong)track.Indexes[0];
2024-05-01 04:05:22 +01:00
if(track.Sequence == 1) track.Indexes[0] = -150;
2024-05-01 04:05:22 +01:00
if(track.Indexes[0] == 0) track.Indexes.Remove(0);
2022-03-06 13:29:38 +00:00
position += 4;
track.StartSector = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\ttrackStart = {0}", track.StartSector);
2022-03-06 13:29:38 +00:00
position += 4;
uint trackLen = BitConverter.ToUInt32(descriptor, position);
2022-03-06 13:29:38 +00:00
// DiscJuggler counts the first track pregap start as 0 instead of -150, we need to adjust appropriately
if(track.StartSector == 0)
trackLen -= 150;
else
track.StartSector -= 150;
int leftLen = (int)trackLen;
2022-03-06 13:29:38 +00:00
// Convert index length to index position
foreach(KeyValuePair<ushort, int> idx in track.Indexes.Reverse())
{
leftLen -= idx.Value;
2024-05-01 04:05:22 +01:00
if(idx.Key == 0 && track.Sequence == 1) continue;
2022-03-06 13:29:38 +00:00
track.Indexes[idx.Key] = (int)track.StartSector + leftLen;
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
track.EndSector = track.StartSector + trackLen - 1;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\ttrackEnd = {0}", track.EndSector);
2022-03-06 13:29:38 +00:00
position += 4;
2022-03-06 13:29:38 +00:00
if(track.Sequence > session.EndTrack)
{
session.EndTrack = track.Sequence;
session.EndSector = track.EndSector;
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
if(track.Sequence < session.StartTrack)
{
session.StartTrack = track.Sequence;
session.StartSector = track.StartSector;
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 16;
2017-12-19 20:33:03 +00:00
uint readMode = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\treadMode = {0}", readMode);
2022-03-06 13:29:38 +00:00
position += 4;
uint trackCtl = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\ttrackCtl = {0}", trackCtl);
2022-03-06 13:29:38 +00:00
position += 4;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 9;
2017-12-19 20:33:03 +00:00
byte[] isrc = new byte[12];
2022-03-06 13:29:38 +00:00
Array.Copy(descriptor, position, isrc, 0, 12);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tisrc = {0}", StringHandlers.CToString(isrc));
2022-03-06 13:29:38 +00:00
position += 12;
uint isrcValid = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tisrc_valid = {0}", isrcValid);
2022-03-06 13:29:38 +00:00
position += 4;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 87;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
byte sessionType = descriptor[position];
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tsessionType = {0}", sessionType);
2022-03-06 13:29:38 +00:00
position++;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 5;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
byte trackFollows = descriptor[position];
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\ttrackFollows = {0}", trackFollows);
2022-03-06 13:29:38 +00:00
position += 2;
2017-12-19 20:33:03 +00:00
uint endAddress = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "\tendAddress = {0}", endAddress);
2022-03-06 13:29:38 +00:00
position += 4;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// As to skip the lead-in
bool firstTrack = currentOffset == 0;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
track.SubchannelType = TrackSubchannelType.None;
2022-03-06 13:29:38 +00:00
switch(trackMode)
{
// Audio
case 0:
2024-05-01 04:05:22 +01:00
if(_imageInfo.SectorSize < 2352) _imageInfo.SectorSize = 2352;
2022-03-06 13:29:38 +00:00
track.Type = TrackType.Audio;
track.BytesPerSector = 2352;
track.RawBytesPerSector = 2352;
2022-03-06 13:29:38 +00:00
switch(readMode)
{
case 2:
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
currentOffset += trackLen * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
break;
case 3:
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.Q16Interleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
break;
case 4:
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 96);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.RawInterleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 96);
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
break;
default:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization.Unknown_read_mode_0, readMode));
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
}
2022-03-06 13:29:38 +00:00
break;
2022-03-06 13:29:38 +00:00
// Mode 1 or DVD
case 1:
2024-05-01 04:05:22 +01:00
if(_imageInfo.SectorSize < 2048) _imageInfo.SectorSize = 2048;
2022-03-06 13:29:38 +00:00
track.Type = TrackType.CdMode1;
track.BytesPerSector = 2048;
2022-03-06 13:29:38 +00:00
switch(readMode)
{
case 0:
track.RawBytesPerSector = 2048;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
currentOffset += trackLen * (ulong)track.RawBytesPerSector;
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
break;
case 1:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization
2024-05-01 04:05:22 +01:00
.Invalid_read_mode_0_for_this_track,
readMode));
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
case 2:
track.RawBytesPerSector = 2352;
currentOffset += trackLen * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEcc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEcc);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccP))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccP);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccQ))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccQ);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
2022-03-06 13:29:38 +00:00
break;
case 3:
track.RawBytesPerSector = 2352;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.Q16Interleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEcc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEcc);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccP))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccP);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccQ))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccQ);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
2022-03-06 13:29:38 +00:00
break;
case 4:
track.RawBytesPerSector = 2352;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 96);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.RawInterleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 96);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEcc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEcc);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccP))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccP);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccQ))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccQ);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
break;
default:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization.Unknown_read_mode_0, readMode));
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
}
2022-03-06 13:29:38 +00:00
break;
2022-03-06 13:29:38 +00:00
// Mode 2
case 2:
2024-05-01 04:05:22 +01:00
if(_imageInfo.SectorSize < 2336) _imageInfo.SectorSize = 2336;
2022-03-06 13:29:38 +00:00
track.Type = TrackType.CdMode2Formless;
track.BytesPerSector = 2336;
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
switch(readMode)
{
case 0:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization
2024-05-01 04:05:22 +01:00
.Invalid_read_mode_0_for_this_track,
readMode));
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
case 1:
track.RawBytesPerSector = 2336;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
currentOffset += trackLen * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
break;
case 2:
track.RawBytesPerSector = 2352;
currentOffset += trackLen * (ulong)track.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2022-03-06 13:29:38 +00:00
break;
case 3:
track.RawBytesPerSector = 2352;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.Q16Interleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 16);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2022-03-06 13:29:38 +00:00
break;
case 4:
track.RawBytesPerSector = 2352;
2024-05-01 04:05:22 +01:00
if(firstTrack) currentOffset += 150 * (ulong)(track.RawBytesPerSector + 96);
2022-03-06 13:29:38 +00:00
track.FileOffset = currentOffset;
track.SubchannelFile = track.File;
track.SubchannelOffset = currentOffset;
track.SubchannelType = TrackSubchannelType.RawInterleaved;
2022-03-06 13:29:38 +00:00
currentOffset += trackLen * (ulong)(track.RawBytesPerSector + 96);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
break;
default:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization.Unknown_read_mode_0, readMode));
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
}
2021-09-21 04:55:28 +01:00
2022-03-06 13:29:38 +00:00
break;
default:
2025-08-17 06:11:22 +01:00
AaruLogging.Error(string.Format(Localization.Unknown_track_mode_0, trackMode));
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.InvalidArgument;
}
2022-03-06 13:29:38 +00:00
track.File = imageFilter.Filename;
track.Filter = imageFilter;
2022-03-06 13:29:38 +00:00
if(track.SubchannelType != TrackSubchannelType.None)
{
track.SubchannelFile = imageFilter.Filename;
track.SubchannelFilter = imageFilter;
2022-03-06 13:29:38 +00:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubchannel))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubchannel);
}
2022-03-06 13:29:38 +00:00
var partition = new Partition
{
Description = track.Description,
Length = trackLen,
Sequence = track.Sequence,
Offset = track.FileOffset,
Start = track.StartSector,
Type = track.Type.ToString()
};
if(track.Sequence > 1 && track.Pregap > 0)
2022-03-06 13:29:38 +00:00
{
partition.Start += track.Pregap;
partition.Length -= track.Pregap;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
partition.Size = partition.Length * (ulong)track.BytesPerSector;
2024-05-01 04:05:22 +01:00
if(track.EndSector + 1 > _imageInfo.Sectors) _imageInfo.Sectors = track.EndSector + 1;
2022-03-06 13:29:38 +00:00
Partitions.Add(partition);
_offsetMap.Add(track.Sequence, track.StartSector);
Tracks.Add(track);
_trackFlags.Add(track.Sequence, (byte)(trackCtl & 0xFF));
addedATrack = true;
2017-12-19 20:33:03 +00:00
}
2024-05-01 04:05:22 +01:00
if(!addedATrack) continue;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
lastSessionTrack = session.EndTrack;
Sessions.Add(session);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "session.StartTrack = {0}", session.StartTrack);
AaruLogging.Debug(MODULE_NAME, "session.StartSector = {0}", session.StartSector);
AaruLogging.Debug(MODULE_NAME, "session.EndTrack = {0}", session.EndTrack);
AaruLogging.Debug(MODULE_NAME, "session.EndSector = {0}", session.EndSector);
2017-12-19 20:33:03 +00:00
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "session.Sequence = {0}", session.Sequence);
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 16;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, Localization.Current_position_equals_0, position);
byte[] filenameB = new byte[descriptor[position]];
2022-03-06 13:29:38 +00:00
position++;
Array.Copy(descriptor, position, filenameB, 0, filenameB.Length);
position += filenameB.Length;
string filename = Path.GetFileName(Encoding.Default.GetString(filenameB));
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "filename = {0}", filename);
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 29;
mediumType = BitConverter.ToUInt16(descriptor, position);
position += 2;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "mediumType = {0}", mediumType);
2022-03-06 13:29:38 +00:00
uint discSize = BitConverter.ToUInt32(descriptor, position);
2022-03-06 13:29:38 +00:00
position += 4;
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "discSize = {0}", discSize);
2022-03-06 13:29:38 +00:00
byte[] volidB = new byte[descriptor[position]];
2022-03-06 13:29:38 +00:00
position++;
Array.Copy(descriptor, position, volidB, 0, volidB.Length);
position += volidB.Length;
string volid = Path.GetFileName(Encoding.Default.GetString(volidB));
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "volid = {0}", volid);
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 9;
byte[] mcn = new byte[13];
2022-03-06 13:29:38 +00:00
Array.Copy(descriptor, position, mcn, 0, 13);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "mcn = {0}", StringHandlers.CToString(mcn));
2022-03-06 13:29:38 +00:00
position += 13;
uint mcnValid = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "mcn_valid = {0}", mcnValid);
2022-03-06 13:29:38 +00:00
position += 4;
uint cdtextLen = BitConverter.ToUInt32(descriptor, position);
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, "cdtextLen = {0}", cdtextLen);
2022-03-06 13:29:38 +00:00
position += 4;
if(cdtextLen > 0)
{
_cdtext = new byte[cdtextLen];
Array.Copy(descriptor, position, _cdtext, 0, cdtextLen);
position += (int)cdtextLen;
_imageInfo.ReadableMediaTags.Add(MediaTagType.CD_TEXT);
}
2022-03-06 13:29:38 +00:00
// Skip unknown
position += 12;
2017-12-19 20:33:03 +00:00
2025-08-17 06:11:22 +01:00
AaruLogging.Debug(MODULE_NAME, Localization.End_position_equals_0, position);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
_imageInfo.MediaType = DecodeCdiMediumType(mediumType);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
if(_imageInfo.MediaType == MediaType.CDROM)
{
bool data = false;
bool mode2 = false;
bool firstaudio = false;
bool firstdata = false;
bool audio = false;
2017-12-19 20:33:03 +00:00
for(int i = 0; i < Tracks.Count; i++)
2022-03-06 13:29:38 +00:00
{
// First track is audio
firstaudio |= i == 0 && Tracks[i].Type == TrackType.Audio;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// First track is data
firstdata |= i == 0 && Tracks[i].Type != TrackType.Audio;
2022-03-06 13:29:38 +00:00
// Any non first track is data
data |= i != 0 && Tracks[i].Type != TrackType.Audio;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
// Any non first track is audio
audio |= i != 0 && Tracks[i].Type == TrackType.Audio;
2017-12-19 20:33:03 +00:00
mode2 = Tracks[i].Type switch
{
TrackType.CdMode2Form1 or TrackType.CdMode2Form2 or TrackType.CdMode2Formless => true,
_ => mode2
};
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
if(!data && !firstdata)
2022-03-06 13:29:38 +00:00
_imageInfo.MediaType = MediaType.CDDA;
else if(firstaudio && data && Sessions.Count > 1 && mode2)
2022-03-06 13:29:38 +00:00
_imageInfo.MediaType = MediaType.CDPLUS;
2023-10-03 23:34:59 +01:00
else if(firstdata && audio || mode2)
2022-03-06 13:29:38 +00:00
_imageInfo.MediaType = MediaType.CDROMXA;
else if(!audio)
_imageInfo.MediaType = MediaType.CDROM;
else
_imageInfo.MediaType = MediaType.CD;
}
if(_trackFlags.Count > 0 && !_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdTrackFlags))
2022-03-06 13:29:38 +00:00
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdTrackFlags);
2022-03-06 13:29:38 +00:00
_imageInfo.Application = "DiscJuggler";
_imageInfo.ImageSize = (ulong)imageFilter.DataForkLength;
_imageInfo.CreationTime = imageFilter.CreationTime;
_imageInfo.LastModificationTime = imageFilter.LastWriteTime;
_imageInfo.MetadataMediaType = MetadataMediaType.OpticalDisc;
2022-03-06 13:29:38 +00:00
_sectorBuilder = new SectorBuilder();
2022-03-06 13:29:38 +00:00
_isCd = mediumType == 152;
2024-05-01 04:05:22 +01:00
if(_isCd) return ErrorNumber.NoError;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
foreach(Track track in Tracks)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
track.Pregap = 0;
track.Indexes?.Clear();
}
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorSync);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorHeader);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorSubHeader);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorEcc);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorEccP);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorEccQ);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdSectorEdc);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdTrackFlags);
_imageInfo.ReadableSectorTags.Remove(SectorTagType.CdTrackIsrc);
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber ReadMediaTag(MediaTagType tag, out byte[] buffer)
{
buffer = null;
2022-03-06 13:29:38 +00:00
switch(tag)
{
case MediaTagType.CD_TEXT:
2017-12-19 20:33:03 +00:00
{
2024-05-01 04:05:22 +01:00
if(_cdtext is { Length: > 0 }) buffer = _cdtext?.Clone() as byte[];
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return buffer != null ? ErrorNumber.NoError : ErrorNumber.NoData;
2017-12-19 20:33:03 +00:00
}
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
2022-03-07 07:36:44 +00:00
public ErrorNumber ReadSector(ulong sectorAddress, out byte[] buffer) => ReadSectors(sectorAddress, 1, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorTag(ulong sectorAddress, SectorTagType tag, out byte[] buffer) =>
ReadSectorsTag(sectorAddress, 1, tag, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSector(ulong sectorAddress, uint track, out byte[] buffer) =>
ReadSectors(sectorAddress, 1, track, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorTag(ulong sectorAddress, uint track, SectorTagType tag, out byte[] buffer) =>
ReadSectorsTag(sectorAddress, 1, track, tag, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectors(ulong sectorAddress, uint length, out byte[] buffer)
{
buffer = null;
2023-10-03 23:34:59 +01:00
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap
where sectorAddress >= kvp.Value
from track in Tracks
where track.Sequence == kvp.Key
2022-03-06 13:29:38 +00:00
where sectorAddress - kvp.Value <
2023-10-03 23:34:59 +01:00
track.EndSector - track.StartSector + 1
select kvp)
2022-03-06 13:29:38 +00:00
return ReadSectors(sectorAddress - kvp.Value, length, kvp.Key, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.SectorNotFound;
}
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, SectorTagType tag, out byte[] buffer)
{
buffer = null;
2017-12-19 20:33:03 +00:00
2023-10-03 23:34:59 +01:00
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap
where sectorAddress >= kvp.Value
from track in Tracks
where track.Sequence == kvp.Key
2022-03-06 13:29:38 +00:00
where sectorAddress - kvp.Value <
2023-10-03 23:34:59 +01:00
track.EndSector - track.StartSector + 1
select kvp)
2022-03-06 13:29:38 +00:00
return ReadSectorsTag(sectorAddress - kvp.Value, length, kvp.Key, tag, out buffer);
2022-03-06 13:29:38 +00:00
return ErrorNumber.SectorNotFound;
}
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectors(ulong sectorAddress, uint length, uint track, out byte[] buffer)
{
buffer = null;
2017-12-19 20:33:03 +00:00
2022-11-15 01:35:06 +00:00
Track aaruTrack = Tracks.FirstOrDefault(linqTrack => linqTrack.Sequence == track);
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(aaruTrack is null) return ErrorNumber.SectorNotFound;
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(length + sectorAddress > aaruTrack.EndSector - aaruTrack.StartSector + 1) return ErrorNumber.OutOfRange;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
uint sectorOffset;
uint sectorSize;
uint sectorSkip;
bool mode2 = false;
2022-03-06 13:29:38 +00:00
switch(aaruTrack.Type)
{
case TrackType.Audio:
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
sectorOffset = 0;
sectorSize = 2352;
sectorSkip = 0;
2022-03-06 13:29:38 +00:00
break;
}
case TrackType.CdMode1:
if(aaruTrack.RawBytesPerSector == 2352)
{
sectorOffset = 16;
sectorSize = 2048;
sectorSkip = 288;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
else
{
sectorOffset = 0;
2022-03-06 13:29:38 +00:00
sectorSize = 2048;
sectorSkip = 0;
}
2022-03-06 13:29:38 +00:00
break;
case TrackType.CdMode2Formless:
{
mode2 = true;
sectorOffset = 0;
sectorSize = (uint)aaruTrack.RawBytesPerSector;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
switch(aaruTrack.SubchannelType)
{
case TrackSubchannelType.None:
sectorSkip += 0;
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.Q16Interleaved:
sectorSkip += 16;
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.RawInterleaved:
sectorSkip += 96;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
buffer = new byte[sectorSize * length];
2023-10-03 23:34:59 +01:00
_imageStream.Seek((long)(aaruTrack.FileOffset + sectorAddress * (sectorOffset + sectorSize + sectorSkip)),
2022-03-06 13:29:38 +00:00
SeekOrigin.Begin);
2022-03-06 13:29:38 +00:00
if(mode2)
{
var mode2Ms = new MemoryStream((int)(sectorSize * length));
2022-03-06 13:29:38 +00:00
buffer = new byte[(aaruTrack.RawBytesPerSector + sectorSkip) * length];
_imageStream.EnsureRead(buffer, 0, buffer.Length);
for(int i = 0; i < length; i++)
2022-03-06 13:29:38 +00:00
{
byte[] sector = new byte[aaruTrack.RawBytesPerSector];
2024-05-01 04:05:22 +01:00
Array.Copy(buffer,
(aaruTrack.RawBytesPerSector + sectorSkip) * i,
sector,
0,
2022-03-06 13:29:38 +00:00
aaruTrack.RawBytesPerSector);
2022-03-06 13:29:38 +00:00
sector = Sector.GetUserDataFromMode2(sector);
mode2Ms.Write(sector, 0, sector.Length);
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
buffer = mode2Ms.ToArray();
2017-12-19 20:33:03 +00:00
}
else if(sectorOffset == 0 && sectorSkip == 0)
_imageStream.EnsureRead(buffer, 0, buffer.Length);
2022-03-06 13:29:38 +00:00
else
2023-10-03 23:34:59 +01:00
{
for(int i = 0; i < length; i++)
2022-03-06 13:29:38 +00:00
{
byte[] sector = new byte[sectorSize];
2022-03-06 13:29:38 +00:00
_imageStream.Seek(sectorOffset, SeekOrigin.Current);
_imageStream.EnsureRead(sector, 0, sector.Length);
2022-03-06 13:29:38 +00:00
_imageStream.Seek(sectorSkip, SeekOrigin.Current);
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
}
2023-10-03 23:34:59 +01:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.NoError;
}
/// <inheritdoc />
2023-10-03 23:34:59 +01:00
public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, uint track, SectorTagType tag,
2022-03-06 13:29:38 +00:00
out byte[] buffer)
{
buffer = null;
2024-05-01 04:05:22 +01:00
if(tag == SectorTagType.CdTrackFlags) track = (uint)sectorAddress;
2022-11-15 01:35:06 +00:00
Track aaruTrack = Tracks.FirstOrDefault(linqTrack => linqTrack.Sequence == track);
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(aaruTrack is null) return ErrorNumber.SectorNotFound;
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(length + sectorAddress > aaruTrack.EndSector - aaruTrack.StartSector + 1) return ErrorNumber.OutOfRange;
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(aaruTrack.Type == TrackType.Data) return ErrorNumber.NotSupported;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
switch(tag)
{
case SectorTagType.CdSectorEcc:
case SectorTagType.CdSectorEccP:
case SectorTagType.CdSectorEccQ:
case SectorTagType.CdSectorEdc:
case SectorTagType.CdSectorHeader:
case SectorTagType.CdSectorSubchannel:
case SectorTagType.CdSectorSubHeader:
2023-10-03 23:34:59 +01:00
case SectorTagType.CdSectorSync:
break;
2022-03-06 13:29:38 +00:00
case SectorTagType.CdTrackFlags:
2024-05-01 04:05:22 +01:00
if(!_trackFlags.TryGetValue(track, out byte flag)) return ErrorNumber.NoData;
2022-03-06 13:29:38 +00:00
2024-05-01 04:39:38 +01:00
buffer = [flag];
2022-03-06 13:29:38 +00:00
return ErrorNumber.NoError;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
uint sectorOffset;
uint sectorSize;
uint sectorSkip;
switch(aaruTrack.Type)
{
case TrackType.CdMode1:
2024-05-01 04:05:22 +01:00
if(aaruTrack.RawBytesPerSector != 2352) return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
switch(tag)
{
case SectorTagType.CdSectorSync:
{
2022-03-06 13:29:38 +00:00
sectorOffset = 0;
sectorSize = 12;
sectorSkip = 2340;
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorHeader:
{
sectorOffset = 12;
sectorSize = 4;
sectorSkip = 2336;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
2023-10-03 23:34:59 +01:00
case SectorTagType.CdSectorSubHeader:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
case SectorTagType.CdSectorEcc:
{
sectorOffset = 2076;
sectorSize = 276;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorEccP:
{
sectorOffset = 2076;
sectorSize = 172;
sectorSkip = 104;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorEccQ:
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
sectorOffset = 2248;
sectorSize = 104;
sectorSkip = 0;
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorEdc:
{
sectorOffset = 2064;
sectorSize = 4;
sectorSkip = 284;
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorSubchannel:
switch(aaruTrack.SubchannelType)
2017-12-19 20:33:03 +00:00
{
2023-10-03 23:34:59 +01:00
case TrackSubchannelType.None:
return ErrorNumber.NoData;
2022-03-06 13:29:38 +00:00
case TrackSubchannelType.Q16Interleaved:
2022-03-06 13:29:38 +00:00
sectorSize = 16;
2022-03-06 13:29:38 +00:00
break;
default:
sectorSize = 96;
2022-03-06 13:29:38 +00:00
break;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
sectorOffset = 2352;
sectorSkip = 0;
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
break;
case TrackType.CdMode2Formless:
2024-05-01 04:05:22 +01:00
if(aaruTrack.RawBytesPerSector != 2352) return ErrorNumber.NotSupported;
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
{
switch(tag)
{
case SectorTagType.CdSectorSync:
case SectorTagType.CdSectorHeader:
case SectorTagType.CdSectorEcc:
case SectorTagType.CdSectorEccP:
2023-10-03 23:34:59 +01:00
case SectorTagType.CdSectorEccQ:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
case SectorTagType.CdSectorSubHeader:
{
sectorOffset = 0;
sectorSize = 8;
sectorSkip = 2328;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
case SectorTagType.CdSectorEdc:
{
sectorOffset = 2332;
sectorSize = 4;
sectorSkip = 0;
2022-03-06 13:29:38 +00:00
break;
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
case SectorTagType.CdSectorSubchannel:
switch(aaruTrack.SubchannelType)
{
2023-10-03 23:34:59 +01:00
case TrackSubchannelType.None:
return ErrorNumber.NoData;
2022-03-06 13:29:38 +00:00
case TrackSubchannelType.Q16Interleaved:
sectorSize = 16;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
default:
sectorSize = 96;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
}
sectorOffset = 2352;
sectorSkip = 0;
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
break;
}
case TrackType.Audio:
{
switch(tag)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
case SectorTagType.CdSectorSubchannel:
switch(aaruTrack.SubchannelType)
2017-12-19 20:33:03 +00:00
{
2023-10-03 23:34:59 +01:00
case TrackSubchannelType.None:
return ErrorNumber.NoData;
2022-03-06 13:29:38 +00:00
case TrackSubchannelType.Q16Interleaved:
sectorSize = 16;
2022-03-06 13:29:38 +00:00
break;
default:
sectorSize = 96;
2022-03-06 13:29:38 +00:00
break;
2017-12-19 20:33:03 +00:00
}
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
sectorOffset = 2352;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
break;
}
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
switch(aaruTrack.SubchannelType)
{
case TrackSubchannelType.None:
sectorSkip += 0;
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.Q16Interleaved:
sectorSkip += 16;
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.RawInterleaved:
sectorSkip += 96;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
buffer = new byte[sectorSize * length];
2017-12-19 20:33:03 +00:00
2023-10-03 23:34:59 +01:00
_imageStream.Seek((long)(aaruTrack.FileOffset + sectorAddress * (sectorOffset + sectorSize + sectorSkip)),
2022-03-06 13:29:38 +00:00
SeekOrigin.Begin);
2017-12-19 20:33:03 +00:00
if(sectorOffset == 0 && sectorSkip == 0)
_imageStream.EnsureRead(buffer, 0, buffer.Length);
2022-03-06 13:29:38 +00:00
else
2023-10-03 23:34:59 +01:00
{
for(int i = 0; i < length; i++)
2017-12-19 20:33:03 +00:00
{
byte[] sector = new byte[sectorSize];
2022-03-06 13:29:38 +00:00
_imageStream.Seek(sectorOffset, SeekOrigin.Current);
_imageStream.EnsureRead(sector, 0, sector.Length);
2022-03-06 13:29:38 +00:00
_imageStream.Seek(sectorSkip, SeekOrigin.Current);
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
}
2023-10-03 23:34:59 +01:00
}
if(aaruTrack.SubchannelType == TrackSubchannelType.Q16Interleaved && tag == SectorTagType.CdSectorSubchannel)
2022-03-06 13:29:38 +00:00
buffer = Subchannel.ConvertQToRaw(buffer);
2022-03-06 13:29:38 +00:00
return ErrorNumber.NoError;
}
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorLong(ulong sectorAddress, out byte[] buffer) =>
ReadSectorsLong(sectorAddress, 1, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorLong(ulong sectorAddress, uint track, out byte[] buffer) =>
ReadSectorsLong(sectorAddress, 1, track, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, out byte[] buffer)
{
buffer = null;
2023-10-03 23:34:59 +01:00
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap
where sectorAddress >= kvp.Value
from track in Tracks
where track.Sequence == kvp.Key
2022-03-06 13:29:38 +00:00
where sectorAddress - kvp.Value <
2023-10-03 23:34:59 +01:00
track.EndSector - track.StartSector + 1
select kvp)
2022-03-06 13:29:38 +00:00
return ReadSectorsLong(sectorAddress - kvp.Value, length, kvp.Key, out buffer);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
return ErrorNumber.SectorNotFound;
}
2020-05-05 15:46:32 +01:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, uint track, out byte[] buffer)
{
buffer = null;
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(!_isCd) return ReadSectors(sectorAddress, length, track, out buffer);
2017-12-19 20:33:03 +00:00
2022-11-15 01:35:06 +00:00
Track aaruTrack = Tracks.FirstOrDefault(linqTrack => linqTrack.Sequence == track);
2017-12-19 20:33:03 +00:00
2024-05-01 04:05:22 +01:00
if(aaruTrack is null) return ErrorNumber.SectorNotFound;
2024-05-01 04:05:22 +01:00
if(length + sectorAddress > aaruTrack.EndSector - aaruTrack.StartSector + 1) return ErrorNumber.OutOfRange;
uint sectorSize = (uint)aaruTrack.RawBytesPerSector;
2022-03-06 13:29:38 +00:00
uint sectorSkip = 0;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
switch(aaruTrack.SubchannelType)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
case TrackSubchannelType.None:
sectorSkip += 0;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.Q16Interleaved:
sectorSkip += 16;
2022-03-06 13:29:38 +00:00
break;
case TrackSubchannelType.RawInterleaved:
sectorSkip += 96;
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
break;
2023-10-03 23:34:59 +01:00
default:
return ErrorNumber.NotSupported;
2022-03-06 13:29:38 +00:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
buffer = new byte[sectorSize * length];
2017-12-19 20:33:03 +00:00
2023-10-03 23:34:59 +01:00
_imageStream.Seek((long)(aaruTrack.FileOffset + sectorAddress * (sectorSize + sectorSkip)), SeekOrigin.Begin);
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
if(sectorSkip == 0)
_imageStream.EnsureRead(buffer, 0, buffer.Length);
2022-03-06 13:29:38 +00:00
else
2023-10-03 23:34:59 +01:00
{
for(int i = 0; i < length; i++)
2017-12-19 20:33:03 +00:00
{
byte[] sector = new byte[sectorSize];
_imageStream.EnsureRead(sector, 0, sector.Length);
2022-03-06 13:29:38 +00:00
_imageStream.Seek(sectorSkip, SeekOrigin.Current);
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
2017-12-19 20:33:03 +00:00
}
2023-10-03 23:34:59 +01:00
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
switch(aaruTrack.Type)
{
case TrackType.CdMode1 when aaruTrack.RawBytesPerSector == 2048:
{
byte[] fullSector = new byte[2352];
byte[] fullBuffer = new byte[2352 * length];
2022-03-06 13:29:38 +00:00
for(uint i = 0; i < length; i++)
2017-12-19 20:33:03 +00:00
{
2022-03-06 13:29:38 +00:00
Array.Copy(buffer, i * 2048, fullSector, 16, 2048);
_sectorBuilder.ReconstructPrefix(ref fullSector, TrackType.CdMode1, (long)(sectorAddress + i));
_sectorBuilder.ReconstructEcc(ref fullSector, TrackType.CdMode1);
Array.Copy(fullSector, 0, fullBuffer, i * 2352, 2352);
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
buffer = fullBuffer;
2022-03-06 13:29:38 +00:00
break;
}
case TrackType.CdMode2Formless when aaruTrack.RawBytesPerSector == 2336:
{
byte[] fullSector = new byte[2352];
byte[] fullBuffer = new byte[2352 * length];
2022-03-06 13:29:38 +00:00
for(uint i = 0; i < length; i++)
{
2024-05-01 04:05:22 +01:00
_sectorBuilder.ReconstructPrefix(ref fullSector,
TrackType.CdMode2Formless,
2022-03-06 13:29:38 +00:00
(long)(sectorAddress + i));
2023-10-03 23:34:59 +01:00
Array.Copy(buffer, i * 2336, fullSector, 16, 2336);
Array.Copy(fullSector, 0, fullBuffer, i * 2352, 2352);
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
buffer = fullBuffer;
2022-03-06 13:29:38 +00:00
break;
}
2017-12-19 20:33:03 +00:00
}
2022-03-06 13:29:38 +00:00
return ErrorNumber.NoError;
}
2017-12-19 20:33:03 +00:00
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
2022-11-13 20:54:01 +00:00
public List<Track> GetSessionTracks(Session session) =>
Sessions.Contains(session) ? GetSessionTracks(session.Sequence) : null;
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public List<Track> GetSessionTracks(ushort session) => Tracks.Where(track => track.Session == session).ToList();
2023-10-03 23:34:59 +01:00
#endregion
2017-12-19 20:33:03 +00:00
}