mirror of
https://github.com/aaru-dps/Aaru.git
synced 2025-12-16 19:24:25 +00:00
1799 lines
63 KiB
C#
1799 lines
63 KiB
C#
// /***************************************************************************
|
|
// Aaru Data Preservation Suite
|
|
// ----------------------------------------------------------------------------
|
|
//
|
|
// Filename : Read.cs
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
|
//
|
|
// Component : Disk image plugins.
|
|
//
|
|
// --[ Description ] ----------------------------------------------------------
|
|
//
|
|
// Reads raw image, that is, user data sector by sector copy.
|
|
//
|
|
// --[ 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/>.
|
|
//
|
|
// ----------------------------------------------------------------------------
|
|
// Copyright © 2011-2025 Natalia Portillo
|
|
// ****************************************************************************/
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using System.Xml.Serialization;
|
|
using Aaru.CommonTypes;
|
|
using Aaru.CommonTypes.AaruMetadata;
|
|
using Aaru.CommonTypes.Enums;
|
|
using Aaru.CommonTypes.Interfaces;
|
|
using Aaru.CommonTypes.Structs.Devices.ATA;
|
|
using Aaru.CommonTypes.Structs.Devices.SCSI;
|
|
using Aaru.Decoders.CD;
|
|
using Aaru.Decoders.DVD;
|
|
using Aaru.Decoders.SCSI;
|
|
using Aaru.Helpers;
|
|
using Aaru.Logging;
|
|
using Schemas;
|
|
using Sentry;
|
|
using DMI = Aaru.Decoders.Xbox.DMI;
|
|
using File = System.IO.File;
|
|
using Inquiry = Aaru.CommonTypes.Structs.Devices.SCSI.Inquiry;
|
|
using Sector = Aaru.Decoders.CD.Sector;
|
|
using Session = Aaru.CommonTypes.Structs.Session;
|
|
using Track = Aaru.CommonTypes.Structs.Track;
|
|
using TrackType = Aaru.CommonTypes.Enums.TrackType;
|
|
|
|
namespace Aaru.Images;
|
|
|
|
public sealed partial class ZZZRawImage
|
|
{
|
|
#region IWritableOpticalImage Members
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber Open(IFilter imageFilter)
|
|
{
|
|
Stream stream = imageFilter.GetDataForkStream();
|
|
stream.Seek(0, SeekOrigin.Begin);
|
|
|
|
_extension = Path.GetExtension(imageFilter.Filename)?.ToLower();
|
|
|
|
switch(_extension)
|
|
{
|
|
case ".1kn":
|
|
_imageInfo.SectorSize = 1024;
|
|
|
|
break;
|
|
case ".2kn":
|
|
_imageInfo.SectorSize = 2048;
|
|
|
|
break;
|
|
case ".4kn":
|
|
_imageInfo.SectorSize = 4096;
|
|
|
|
break;
|
|
case ".8kn":
|
|
_imageInfo.SectorSize = 8192;
|
|
|
|
break;
|
|
case ".16kn":
|
|
_imageInfo.SectorSize = 16384;
|
|
|
|
break;
|
|
case ".32kn":
|
|
_imageInfo.SectorSize = 32768;
|
|
|
|
break;
|
|
case ".64kn":
|
|
_imageInfo.SectorSize = 65536;
|
|
|
|
break;
|
|
case ".512":
|
|
case ".512e":
|
|
_imageInfo.SectorSize = 512;
|
|
|
|
break;
|
|
case ".128":
|
|
_imageInfo.SectorSize = 128;
|
|
|
|
break;
|
|
case ".256":
|
|
_imageInfo.SectorSize = 256;
|
|
|
|
break;
|
|
case ".2352":
|
|
_imageInfo.SectorSize = 2352;
|
|
|
|
break;
|
|
case ".2448":
|
|
_imageInfo.SectorSize = 2448;
|
|
|
|
break;
|
|
|
|
case ".iso" when imageFilter.DataForkLength % 2048 == 0:
|
|
_imageInfo.SectorSize = 2048;
|
|
|
|
break;
|
|
case ".toast" when imageFilter.DataForkLength % 2048 == 0:
|
|
_imageInfo.SectorSize = 2048;
|
|
|
|
break;
|
|
case ".toast" when imageFilter.DataForkLength % 2336 == 0:
|
|
_imageInfo.SectorSize = 2336;
|
|
|
|
break;
|
|
case ".toast" when imageFilter.DataForkLength % 2352 == 0:
|
|
_imageInfo.SectorSize = 2352;
|
|
|
|
break;
|
|
case ".toast" when imageFilter.DataForkLength % 2056 == 0:
|
|
_imageInfo.SectorSize = 2352;
|
|
_toastXa = true;
|
|
|
|
break;
|
|
case ".d81" when imageFilter.DataForkLength == 819200:
|
|
_imageInfo.SectorSize = 256;
|
|
|
|
break;
|
|
case ".raw" when imageFilter.DataForkLength % 2064 == 0:
|
|
_imageInfo.SectorSize = 2048;
|
|
_rawDvd = true;
|
|
|
|
break;
|
|
default:
|
|
switch(_extension)
|
|
{
|
|
case ".adf" or ".adl" or ".ssd" or ".dsd"
|
|
when imageFilter.DataForkLength is 163840 or 327680 or 655360:
|
|
_imageInfo.SectorSize = 256;
|
|
|
|
break;
|
|
case ".adf" or ".adl" when imageFilter.DataForkLength == 819200:
|
|
_imageInfo.SectorSize = 1024;
|
|
|
|
break;
|
|
default:
|
|
switch(imageFilter.DataForkLength)
|
|
{
|
|
case 242944:
|
|
case 256256:
|
|
case 495872:
|
|
case 92160:
|
|
case 133120:
|
|
_imageInfo.SectorSize = 128;
|
|
|
|
break;
|
|
case 116480:
|
|
case 287488: // T0S0 = 128bps
|
|
case 988416: // T0S0 = 128bps
|
|
case 995072: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 1021696: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 232960:
|
|
case 143360:
|
|
case 286720:
|
|
case 512512:
|
|
case 102400:
|
|
case 204800:
|
|
case 655360:
|
|
case 80384: // T0S0 = 128bps
|
|
case 325632: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 653312: // T0S0 = 128bps, T0S1 = 256bps
|
|
|
|
#region Commodore
|
|
|
|
case 174848:
|
|
case 175531:
|
|
case 196608:
|
|
case 197376:
|
|
case 349696:
|
|
case 351062:
|
|
case 822400:
|
|
|
|
#endregion Commodore
|
|
|
|
_imageInfo.SectorSize = 256;
|
|
|
|
break;
|
|
case 81664:
|
|
_imageInfo.SectorSize = 319;
|
|
|
|
break;
|
|
case 306432: // T0S0 = 128bps
|
|
case 1146624: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 1177344: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.SectorSize = 512;
|
|
|
|
break;
|
|
case 1222400: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 1304320: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 1255168: // T0S0 = 128bps, T0S1 = 256bps
|
|
case 1261568:
|
|
case 1638400:
|
|
_imageInfo.SectorSize = 1024;
|
|
|
|
break;
|
|
case 5013504: // SyQuest SQ100 cartridge
|
|
case 10076160: // SyQuest SQ200 cartridge
|
|
_imageInfo.SectorSize = 256;
|
|
|
|
break;
|
|
case 35002122240:
|
|
_imageInfo.SectorSize = 2048;
|
|
|
|
break;
|
|
default:
|
|
_imageInfo.SectorSize = 512;
|
|
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
_imageInfo.ImageSize = (ulong)imageFilter.DataForkLength;
|
|
_imageInfo.CreationTime = imageFilter.CreationTime;
|
|
_imageInfo.LastModificationTime = imageFilter.LastWriteTime;
|
|
_imageInfo.MediaTitle = Path.GetFileNameWithoutExtension(imageFilter.Filename);
|
|
_differentTrackZeroSize = false;
|
|
_rawImageFilter = imageFilter;
|
|
|
|
switch(imageFilter.DataForkLength)
|
|
{
|
|
case 242944:
|
|
_imageInfo.Sectors = 1898;
|
|
|
|
break;
|
|
case 256256:
|
|
_imageInfo.Sectors = 2002;
|
|
|
|
break;
|
|
case 495872:
|
|
_imageInfo.Sectors = 3874;
|
|
|
|
break;
|
|
case 116480:
|
|
_imageInfo.Sectors = 455;
|
|
|
|
break;
|
|
case 287488: // T0S0 = 128bps
|
|
_imageInfo.Sectors = 1136;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 988416: // T0S0 = 128bps
|
|
_imageInfo.Sectors = 3874;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 995072: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 3900;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1021696: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 4004;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 81664:
|
|
_imageInfo.Sectors = 256;
|
|
|
|
break;
|
|
case 306432: // T0S0 = 128bps
|
|
_imageInfo.Sectors = 618;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1146624: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 2272;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1177344: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 2332;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1222400: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 1236;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1304320: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 1316;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1255168: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 1268;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 80384: // T0S0 = 128bps
|
|
_imageInfo.Sectors = 322;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 325632: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 1280;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 653312: // T0S0 = 128bps, T0S1 = 256bps
|
|
_imageInfo.Sectors = 2560;
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 1880064: // IBM XDF, 3,5", real number of sectors
|
|
_imageInfo.Sectors = 670;
|
|
_imageInfo.SectorSize = 8192; // Biggest sector size
|
|
_differentTrackZeroSize = true;
|
|
|
|
break;
|
|
case 175531:
|
|
_imageInfo.Sectors = 683;
|
|
|
|
break;
|
|
case 197375:
|
|
_imageInfo.Sectors = 768;
|
|
|
|
break;
|
|
case 351062:
|
|
_imageInfo.Sectors = 1366;
|
|
|
|
break;
|
|
case 822400:
|
|
_imageInfo.Sectors = 3200;
|
|
|
|
break;
|
|
default:
|
|
_imageInfo.Sectors = _imageInfo.ImageSize / _imageInfo.SectorSize;
|
|
|
|
break;
|
|
}
|
|
|
|
_imageInfo.MediaType = CalculateDiskType();
|
|
|
|
if(_imageInfo.ImageSize % 2352 == 0 || _imageInfo.ImageSize % 2448 == 0)
|
|
{
|
|
byte[] sync = new byte[12];
|
|
byte[] header = new byte[4];
|
|
stream.Seek(0, SeekOrigin.Begin);
|
|
stream.EnsureRead(sync, 0, 12);
|
|
stream.EnsureRead(header, 0, 4);
|
|
|
|
if(_cdSync.SequenceEqual(sync))
|
|
{
|
|
_rawCompactDisc = true;
|
|
_hasSubchannel = _imageInfo.ImageSize % 2448 == 0;
|
|
_imageInfo.Sectors = _imageInfo.ImageSize / (ulong)(_hasSubchannel ? 2448 : 2352);
|
|
_imageInfo.MediaType = MediaType.CD;
|
|
_mode2 = header[3] == 0x02;
|
|
_imageInfo.SectorSize = (uint)(_mode2 ? 2336 : 2048);
|
|
}
|
|
}
|
|
|
|
if(_toastXa) _imageInfo.MediaType = MediaType.CD;
|
|
|
|
if(_rawDvd) _imageInfo.Sectors = _imageInfo.ImageSize / 2064;
|
|
|
|
// Sharp X68000 SASI hard disks
|
|
if(_extension == ".hdf")
|
|
{
|
|
if(_imageInfo.ImageSize % 256 == 0)
|
|
{
|
|
_imageInfo.SectorSize = 256;
|
|
_imageInfo.Sectors = _imageInfo.ImageSize / _imageInfo.SectorSize;
|
|
_imageInfo.MediaType = MediaType.GENERIC_HDD;
|
|
}
|
|
}
|
|
|
|
// Search for known tags
|
|
string basename = imageFilter.BasePath;
|
|
basename = basename[..^(_extension?.Length ?? basename.Length)];
|
|
|
|
_mediaTags = new Dictionary<MediaTagType, byte[]>();
|
|
|
|
foreach((MediaTagType tag, string name) sidecar in _readWriteSidecars)
|
|
{
|
|
try
|
|
{
|
|
IFilter filter = PluginRegister.Singleton.GetFilter(basename + sidecar.name);
|
|
|
|
if(filter is null) continue;
|
|
|
|
AaruLogging.Debug(MODULE_NAME, Localization.Found_media_tag_0, sidecar.tag);
|
|
byte[] data = new byte[filter.DataForkLength];
|
|
filter.GetDataForkStream().EnsureRead(data, 0, data.Length);
|
|
_mediaTags.Add(sidecar.tag, data);
|
|
}
|
|
catch(IOException ex)
|
|
{
|
|
SentrySdk.CaptureException(ex);
|
|
}
|
|
}
|
|
|
|
// If there are INQUIRY and IDENTIFY tags, it's ATAPI
|
|
if(_mediaTags.ContainsKey(MediaTagType.SCSI_INQUIRY))
|
|
{
|
|
if(_mediaTags.TryGetValue(MediaTagType.ATA_IDENTIFY, out byte[] tag))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.ATA_IDENTIFY);
|
|
_mediaTags.Add(MediaTagType.ATAPI_IDENTIFY, tag);
|
|
}
|
|
}
|
|
|
|
// It is a blu-ray
|
|
if(_mediaTags.ContainsKey(MediaTagType.BD_DI))
|
|
{
|
|
_imageInfo.MediaType = MediaType.BDROM;
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.DVD_BCA, out byte[] bca))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.DVD_BCA);
|
|
_mediaTags.Add(MediaTagType.BD_BCA, bca);
|
|
}
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.DVDRAM_DDS, out byte[] dds))
|
|
{
|
|
_imageInfo.MediaType = MediaType.BDRE;
|
|
_mediaTags.Remove(MediaTagType.DVDRAM_DDS);
|
|
_mediaTags.Add(MediaTagType.BD_DDS, dds);
|
|
}
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.DVDRAM_SpareArea, out byte[] sai))
|
|
{
|
|
_imageInfo.MediaType = MediaType.BDRE;
|
|
_mediaTags.Remove(MediaTagType.DVDRAM_SpareArea);
|
|
_mediaTags.Add(MediaTagType.BD_SpareArea, sai);
|
|
}
|
|
}
|
|
|
|
// It is a DVD
|
|
if(_mediaTags.TryGetValue(MediaTagType.DVD_PFI, out byte[] pfi))
|
|
{
|
|
PFI.PhysicalFormatInformation decPfi = PFI.Decode(pfi, _imageInfo.MediaType).Value;
|
|
|
|
_imageInfo.MediaType = decPfi.DiskCategory switch
|
|
{
|
|
DiskCategory.DVDPR => MediaType.DVDPR,
|
|
DiskCategory.DVDPRDL => MediaType.DVDPRDL,
|
|
DiskCategory.DVDPRW => MediaType.DVDPRW,
|
|
DiskCategory.DVDPRWDL => MediaType.DVDPRWDL,
|
|
DiskCategory.DVDR => decPfi.PartVersion >= 6 ? MediaType.DVDRDL : MediaType.DVDR,
|
|
DiskCategory.DVDRAM => MediaType.DVDRAM,
|
|
DiskCategory.DVDRW => decPfi.PartVersion >= 15
|
|
? MediaType.DVDRWDL
|
|
: MediaType.DVDRW,
|
|
DiskCategory.HDDVDR => MediaType.HDDVDR,
|
|
DiskCategory.HDDVDRAM => MediaType.HDDVDRAM,
|
|
DiskCategory.HDDVDROM => MediaType.HDDVDROM,
|
|
DiskCategory.HDDVDRW => MediaType.HDDVDRW,
|
|
DiskCategory.Nintendo => decPfi.DiscSize == DVDSize.Eighty
|
|
? MediaType.GOD
|
|
: MediaType.WOD,
|
|
DiskCategory.UMD => MediaType.UMD,
|
|
_ => MediaType.DVDROM
|
|
};
|
|
|
|
if(_imageInfo.MediaType is MediaType.DVDR or MediaType.DVDRW or MediaType.HDDVDR &&
|
|
_mediaTags.TryGetValue(MediaTagType.DVD_MediaIdentifier, out byte[] mid))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.DVD_MediaIdentifier);
|
|
_mediaTags.Add(MediaTagType.DVDR_MediaIdentifier, mid);
|
|
}
|
|
|
|
// Check for Xbox
|
|
if(_mediaTags.TryGetValue(MediaTagType.DVD_DMI, out byte[] dmi))
|
|
{
|
|
if(DMI.IsXbox(dmi) || DMI.IsXbox360(dmi))
|
|
{
|
|
if(DMI.IsXbox(dmi))
|
|
_imageInfo.MediaType = MediaType.XGD;
|
|
else if(DMI.IsXbox360(dmi))
|
|
{
|
|
_imageInfo.MediaType = MediaType.XGD2;
|
|
|
|
// All XGD3 all have the same number of blocks
|
|
if(_imageInfo.Sectors is 25063 or 4229664 or 4246304) // Wxripper unlock
|
|
_imageInfo.MediaType = MediaType.XGD3;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// It's MultiMediaCard or SecureDigital
|
|
if(_mediaTags.ContainsKey(MediaTagType.SD_CID) ||
|
|
_mediaTags.ContainsKey(MediaTagType.SD_CSD) ||
|
|
_mediaTags.ContainsKey(MediaTagType.SD_OCR))
|
|
{
|
|
_imageInfo.MediaType = MediaType.SecureDigital;
|
|
|
|
if(_mediaTags.ContainsKey(MediaTagType.MMC_ExtendedCSD) || !_mediaTags.ContainsKey(MediaTagType.SD_SCR))
|
|
{
|
|
_imageInfo.MediaType = MediaType.MMC;
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.SD_CID, out byte[] cid))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.SD_CID);
|
|
_mediaTags.Add(MediaTagType.MMC_CID, cid);
|
|
}
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.SD_CSD, out byte[] csd))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.SD_CSD);
|
|
_mediaTags.Add(MediaTagType.MMC_CSD, csd);
|
|
}
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.SD_OCR, out byte[] ocr))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.SD_OCR);
|
|
_mediaTags.Add(MediaTagType.MMC_OCR, ocr);
|
|
}
|
|
}
|
|
}
|
|
|
|
// It's a compact disc
|
|
if(_mediaTags.ContainsKey(MediaTagType.CD_FullTOC))
|
|
{
|
|
_imageInfo.MediaType = _imageInfo.Sectors > 360000 ? MediaType.DDCD : MediaType.CD;
|
|
|
|
// Only CD-R and CD-RW have ATIP
|
|
if(_mediaTags.TryGetValue(MediaTagType.CD_ATIP, out byte[] atipBuf))
|
|
{
|
|
ATIP.CDATIP atip = ATIP.Decode(atipBuf);
|
|
|
|
if(atip != null) _imageInfo.MediaType = atip.DiscType ? MediaType.CDRW : MediaType.CDR;
|
|
}
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.Floppy_LeadOut, out byte[] leadout))
|
|
{
|
|
_mediaTags.Remove(MediaTagType.Floppy_LeadOut);
|
|
_mediaTags.Add(MediaTagType.CD_LeadOut, leadout);
|
|
}
|
|
}
|
|
|
|
switch(_imageInfo.MediaType)
|
|
{
|
|
case MediaType.ACORN_35_DS_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 5;
|
|
|
|
break;
|
|
case MediaType.ACORN_35_DS_HD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.ACORN_525_DS_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.ACORN_525_SS_DD_40:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.ACORN_525_SS_DD_80:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.ACORN_525_SS_SD_40:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.ACORN_525_SS_SD_80:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.Apple32DS:
|
|
_imageInfo.Cylinders = 35;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 13;
|
|
|
|
break;
|
|
case MediaType.Apple32SS:
|
|
_imageInfo.Cylinders = 36;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 13;
|
|
|
|
break;
|
|
case MediaType.Apple33DS:
|
|
_imageInfo.Cylinders = 35;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.Apple33SS:
|
|
_imageInfo.Cylinders = 35;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.AppleSonyDS:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.AppleSonySS:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.ATARI_35_DS_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.ATARI_35_DS_DD_11:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 11;
|
|
|
|
break;
|
|
case MediaType.ATARI_35_SS_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.ATARI_35_SS_DD_11:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 11;
|
|
|
|
break;
|
|
case MediaType.ATARI_525_ED:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.ATARI_525_SD:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 18;
|
|
|
|
break;
|
|
case MediaType.CBM_35_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.CBM_AMIGA_35_DD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 11;
|
|
|
|
break;
|
|
case MediaType.CBM_AMIGA_35_HD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 22;
|
|
|
|
break;
|
|
case MediaType.DMF:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 21;
|
|
|
|
break;
|
|
case MediaType.DOS_35_DS_DD_9:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.Apricot_35:
|
|
_imageInfo.Cylinders = 70;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.DOS_35_ED:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 36;
|
|
|
|
break;
|
|
case MediaType.DOS_35_HD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 18;
|
|
|
|
break;
|
|
case MediaType.DOS_35_SS_DD_9:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.DOS_525_DS_DD_8:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.DOS_525_DS_DD_9:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.DOS_525_HD:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 15;
|
|
|
|
break;
|
|
case MediaType.DOS_525_SS_DD_8:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.DOS_525_SS_DD_9:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.ECMA_54:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.ECMA_59:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.ECMA_66:
|
|
_imageInfo.Cylinders = 35;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 9;
|
|
|
|
break;
|
|
case MediaType.ECMA_69_8:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.ECMA_70:
|
|
_imageInfo.Cylinders = 40;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.ECMA_78:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.ECMA_99_15:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 15;
|
|
|
|
break;
|
|
case MediaType.ECMA_99_26:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.ECMA_99_8:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.FDFORMAT_35_DD:
|
|
_imageInfo.Cylinders = 82;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 10;
|
|
|
|
break;
|
|
case MediaType.FDFORMAT_35_HD:
|
|
_imageInfo.Cylinders = 82;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 21;
|
|
|
|
break;
|
|
case MediaType.FDFORMAT_525_HD:
|
|
_imageInfo.Cylinders = 82;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 17;
|
|
|
|
break;
|
|
case MediaType.IBM23FD:
|
|
_imageInfo.Cylinders = 32;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.IBM33FD_128:
|
|
_imageInfo.Cylinders = 73;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.IBM33FD_256:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 15;
|
|
|
|
break;
|
|
case MediaType.IBM33FD_512:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.IBM43FD_128:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.IBM43FD_256:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 15;
|
|
|
|
break;
|
|
case MediaType.IBM53FD_1024:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.IBM53FD_256:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 26;
|
|
|
|
break;
|
|
case MediaType.IBM53FD_512:
|
|
_imageInfo.Cylinders = 74;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 15;
|
|
|
|
break;
|
|
case MediaType.NEC_35_TD:
|
|
_imageInfo.Cylinders = 240;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 38;
|
|
|
|
break;
|
|
case MediaType.NEC_525_HD:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 8;
|
|
|
|
break;
|
|
case MediaType.XDF_35:
|
|
_imageInfo.Cylinders = 80;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 23;
|
|
|
|
break;
|
|
case MediaType.SQ100:
|
|
_imageInfo.Cylinders = 306;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.SQ200:
|
|
_imageInfo.Cylinders = 615;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
|
|
// Following ones are what the device itself report, not the physical geometry
|
|
case MediaType.Jaz:
|
|
_imageInfo.Cylinders = 1021;
|
|
_imageInfo.Heads = 64;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.PocketZip:
|
|
_imageInfo.Cylinders = 154;
|
|
_imageInfo.Heads = 16;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.LS120:
|
|
_imageInfo.Cylinders = 963;
|
|
_imageInfo.Heads = 8;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.LS240:
|
|
_imageInfo.Cylinders = 262;
|
|
_imageInfo.Heads = 32;
|
|
_imageInfo.SectorsPerTrack = 56;
|
|
|
|
break;
|
|
case MediaType.FD32MB:
|
|
_imageInfo.Cylinders = 1024;
|
|
_imageInfo.Heads = 2;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.ZIP100:
|
|
_imageInfo.Cylinders = 96;
|
|
_imageInfo.Heads = 64;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.ZIP250:
|
|
_imageInfo.Cylinders = 239;
|
|
_imageInfo.Heads = 64;
|
|
_imageInfo.SectorsPerTrack = 32;
|
|
|
|
break;
|
|
case MediaType.MetaFloppy_Mod_I:
|
|
_imageInfo.Cylinders = 35;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
case MediaType.MetaFloppy_Mod_II:
|
|
_imageInfo.Cylinders = 77;
|
|
_imageInfo.Heads = 1;
|
|
_imageInfo.SectorsPerTrack = 16;
|
|
|
|
break;
|
|
default:
|
|
_imageInfo.Cylinders = (uint)(_imageInfo.Sectors / 16 / 63);
|
|
_imageInfo.Heads = 16;
|
|
_imageInfo.SectorsPerTrack = 63;
|
|
|
|
break;
|
|
}
|
|
|
|
// It's SCSI, check tags
|
|
if(_mediaTags.ContainsKey(MediaTagType.SCSI_INQUIRY))
|
|
{
|
|
PeripheralDeviceTypes devType = PeripheralDeviceTypes.DirectAccess;
|
|
Inquiry? scsiInq = null;
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.SCSI_INQUIRY, out byte[] inq))
|
|
{
|
|
scsiInq = Inquiry.Decode(inq);
|
|
devType = (PeripheralDeviceTypes)(inq[0] & 0x1F);
|
|
}
|
|
|
|
Modes.DecodedMode? decMode = null;
|
|
|
|
if(_mediaTags.TryGetValue(MediaTagType.SCSI_MODESENSE_10, out byte[] mode10))
|
|
decMode = Modes.DecodeMode10(mode10, devType);
|
|
else if(_mediaTags.TryGetValue(MediaTagType.SCSI_MODESENSE_6, out byte[] mode6))
|
|
decMode = Modes.DecodeMode6(mode6, devType);
|
|
|
|
byte mediumType = 0;
|
|
byte densityCode = 0;
|
|
|
|
if(decMode.HasValue)
|
|
{
|
|
mediumType = (byte)decMode.Value.Header.MediumType;
|
|
|
|
if(decMode?.Header.BlockDescriptors?.Length > 0)
|
|
densityCode = (byte)decMode.Value.Header.BlockDescriptors[0].Density;
|
|
|
|
if(decMode.Value.Pages != null)
|
|
{
|
|
foreach(Modes.ModePage page in decMode.Value.Pages)
|
|
|
|
{
|
|
switch(page.Page)
|
|
{
|
|
// CD-ROM page
|
|
case 0x2A when page.Subpage == 0:
|
|
{
|
|
if(_mediaTags.ContainsKey(MediaTagType.SCSI_MODEPAGE_2A))
|
|
_mediaTags.Remove(MediaTagType.SCSI_MODEPAGE_2A);
|
|
|
|
_mediaTags.Add(MediaTagType.SCSI_MODEPAGE_2A, page.PageResponse);
|
|
|
|
break;
|
|
}
|
|
|
|
// Rigid Disk page
|
|
case 0x04 when page.Subpage == 0:
|
|
{
|
|
Modes.ModePage_04? mode04 = Modes.DecodeModePage_04(page.PageResponse);
|
|
|
|
if(!mode04.HasValue) continue;
|
|
|
|
_imageInfo.Cylinders = mode04.Value.Cylinders;
|
|
_imageInfo.Heads = mode04.Value.Heads;
|
|
|
|
_imageInfo.SectorsPerTrack =
|
|
(uint)(_imageInfo.Sectors / (mode04.Value.Cylinders * mode04.Value.Heads));
|
|
|
|
break;
|
|
}
|
|
|
|
// Flexible Disk Page
|
|
case 0x05 when page.Subpage == 0:
|
|
{
|
|
Modes.ModePage_05? mode05 = Modes.DecodeModePage_05(page.PageResponse);
|
|
|
|
if(!mode05.HasValue) continue;
|
|
|
|
_imageInfo.Cylinders = mode05.Value.Cylinders;
|
|
_imageInfo.Heads = mode05.Value.Heads;
|
|
_imageInfo.SectorsPerTrack = mode05.Value.SectorsPerTrack;
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(scsiInq.HasValue)
|
|
{
|
|
_imageInfo.DriveManufacturer =
|
|
VendorString.Prettify(StringHandlers.CToString(scsiInq.Value.VendorIdentification).Trim());
|
|
|
|
_imageInfo.DriveModel = StringHandlers.CToString(scsiInq.Value.ProductIdentification).Trim();
|
|
|
|
_imageInfo.DriveFirmwareRevision = StringHandlers.CToString(scsiInq.Value.ProductRevisionLevel).Trim();
|
|
|
|
_imageInfo.MediaType = MediaTypeFromDevice.GetFromScsi((byte)devType,
|
|
_imageInfo.DriveManufacturer,
|
|
_imageInfo.DriveModel,
|
|
mediumType,
|
|
densityCode,
|
|
_imageInfo.Sectors,
|
|
_imageInfo.SectorSize,
|
|
_mediaTags.ContainsKey(MediaTagType
|
|
.USB_Descriptors),
|
|
_rawCompactDisc);
|
|
}
|
|
|
|
if(_imageInfo.MediaType == MediaType.Unknown)
|
|
{
|
|
_imageInfo.MediaType = devType == PeripheralDeviceTypes.OpticalDevice
|
|
? MediaType.UnknownMO
|
|
: MediaType.GENERIC_HDD;
|
|
}
|
|
}
|
|
|
|
// It's ATA, check tags
|
|
if(_mediaTags.TryGetValue(MediaTagType.ATA_IDENTIFY, out byte[] identifyBuf))
|
|
{
|
|
Identify.IdentifyDevice? ataId = CommonTypes.Structs.Devices.ATA.Identify.Decode(identifyBuf);
|
|
|
|
if(ataId.HasValue)
|
|
{
|
|
_imageInfo.MediaType = (ushort)ataId.Value.GeneralConfiguration == 0x848A
|
|
? MediaType.CompactFlash
|
|
: MediaType.GENERIC_HDD;
|
|
|
|
if(ataId.Value.Cylinders == 0 || ataId.Value.Heads == 0 || ataId.Value.SectorsPerTrack == 0)
|
|
{
|
|
_imageInfo.Cylinders = ataId.Value.CurrentCylinders;
|
|
_imageInfo.Heads = ataId.Value.CurrentHeads;
|
|
_imageInfo.SectorsPerTrack = ataId.Value.CurrentSectorsPerTrack;
|
|
}
|
|
else
|
|
{
|
|
_imageInfo.Cylinders = ataId.Value.Cylinders;
|
|
_imageInfo.Heads = ataId.Value.Heads;
|
|
_imageInfo.SectorsPerTrack = ataId.Value.SectorsPerTrack;
|
|
}
|
|
}
|
|
}
|
|
|
|
switch(_imageInfo.MediaType)
|
|
{
|
|
case MediaType.CD:
|
|
case MediaType.CDRW:
|
|
case MediaType.CDR:
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdTrackFlags))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdTrackFlags);
|
|
|
|
goto case MediaType.BDRE;
|
|
case MediaType.BDRE:
|
|
case MediaType.BDROM:
|
|
case MediaType.BDR:
|
|
case MediaType.BDRXL:
|
|
case MediaType.DVDPR:
|
|
case MediaType.DVDPRDL:
|
|
case MediaType.DVDPRW:
|
|
case MediaType.DVDPRWDL:
|
|
case MediaType.DVDRDL:
|
|
case MediaType.DVDR:
|
|
case MediaType.DVDRAM:
|
|
case MediaType.DVDROM:
|
|
case MediaType.DVDRWDL:
|
|
case MediaType.DVDRW:
|
|
case MediaType.HDDVDR:
|
|
case MediaType.HDDVDRAM:
|
|
case MediaType.HDDVDROM:
|
|
case MediaType.HDDVDRW:
|
|
case MediaType.GOD:
|
|
case MediaType.WOD:
|
|
case MediaType.UMD:
|
|
case MediaType.XGD:
|
|
case MediaType.XGD2:
|
|
case MediaType.XGD3:
|
|
case MediaType.PD650:
|
|
case MediaType.PD650_WORM:
|
|
_imageInfo.MetadataMediaType = MetadataMediaType.OpticalDisc;
|
|
|
|
break;
|
|
default:
|
|
_imageInfo.MetadataMediaType = MetadataMediaType.BlockMedia;
|
|
|
|
break;
|
|
}
|
|
|
|
if(_imageInfo.MetadataMediaType == MetadataMediaType.OpticalDisc)
|
|
{
|
|
_imageInfo.HasSessions = true;
|
|
_imageInfo.HasPartitions = true;
|
|
}
|
|
|
|
AaruLogging.Verbose(Localization.Raw_disk_image_contains_a_disk_of_type_0, _imageInfo.MediaType);
|
|
|
|
try
|
|
{
|
|
if(File.Exists(basename + ".metadata.json"))
|
|
{
|
|
var fs = new FileStream(basename + ".metadata.json", FileMode.Open);
|
|
|
|
AaruMetadata =
|
|
(JsonSerializer.Deserialize(fs, typeof(MetadataJson), MetadataJsonContext.Default) as MetadataJson)
|
|
?.AaruMetadata;
|
|
|
|
fs.Close();
|
|
}
|
|
else if(File.Exists(basename + ".cicm.xml"))
|
|
{
|
|
// The converter to AaruMetadata basically overcomes this (should?)
|
|
#pragma warning disable IL2026
|
|
var sidecarXs = new XmlSerializer(typeof(CICMMetadataType));
|
|
#pragma warning restore IL2026
|
|
|
|
var sr = new StreamReader(basename + ".cicm.xml");
|
|
|
|
// The converter to AaruMetadata basically overcomes this (should?)
|
|
#pragma warning disable IL2026
|
|
AaruMetadata = (CICMMetadataType)sidecarXs.Deserialize(sr);
|
|
#pragma warning restore IL2026
|
|
|
|
sr.Close();
|
|
}
|
|
}
|
|
catch(Exception ex)
|
|
{
|
|
// Do nothing.
|
|
SentrySdk.CaptureException(ex);
|
|
}
|
|
|
|
_imageInfo.ReadableMediaTags = [.._mediaTags.Keys];
|
|
|
|
if(_rawDvd)
|
|
{
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorInformation))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorInformation);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorNumber))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorNumber);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorIed))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorIed);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorCmi))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorCmi);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorTitleKey))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorTitleKey);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorEdc))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorEdc);
|
|
}
|
|
|
|
if(!_rawCompactDisc && !_toastXa) return ErrorNumber.NoError;
|
|
|
|
if(_hasSubchannel)
|
|
{
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubchannel))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubchannel);
|
|
}
|
|
|
|
if(_mode2 || _toastXa)
|
|
{
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubHeader))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubHeader);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
|
|
}
|
|
else
|
|
{
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubHeader))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubHeader);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEcc))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEcc);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccP))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccP);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccQ))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccQ);
|
|
|
|
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
|
|
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
|
|
}
|
|
|
|
return ErrorNumber.NoError;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSector(ulong sectorAddress, out byte[] buffer) => ReadSectors(sectorAddress, 1, out buffer);
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectors(ulong sectorAddress, uint length, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_differentTrackZeroSize) return ErrorNumber.NotImplemented;
|
|
|
|
if(sectorAddress > _imageInfo.Sectors - 1) return ErrorNumber.OutOfRange;
|
|
|
|
if(sectorAddress + length > _imageInfo.Sectors) return ErrorNumber.OutOfRange;
|
|
|
|
Stream stream = _rawImageFilter.GetDataForkStream();
|
|
|
|
uint sectorOffset = 0;
|
|
uint sectorSize = _imageInfo.SectorSize;
|
|
uint sectorSkip = 0;
|
|
|
|
if(_rawCompactDisc)
|
|
{
|
|
sectorOffset = (uint)(_mode2 ? 0 : 16);
|
|
sectorSize = (uint)(_mode2 ? 2352 : 2048);
|
|
sectorSkip = (uint)(_mode2 ? 0 : 288);
|
|
}
|
|
|
|
// TODO: Handle 2336 bps images
|
|
if(_toastXa)
|
|
{
|
|
sectorOffset = 8;
|
|
sectorSize = 2048;
|
|
sectorSkip = 0;
|
|
}
|
|
|
|
if(_hasSubchannel) sectorSkip += 96;
|
|
|
|
if(_rawDvd)
|
|
{
|
|
sectorOffset = 12;
|
|
sectorSize = 2048;
|
|
sectorSkip = 4;
|
|
}
|
|
|
|
buffer = new byte[sectorSize * length];
|
|
|
|
var br = new BinaryReader(stream);
|
|
br.BaseStream.Seek((long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip)), SeekOrigin.Begin);
|
|
|
|
if(_mode2)
|
|
{
|
|
var mode2Ms = new MemoryStream((int)(sectorSize * length));
|
|
|
|
buffer = br.ReadBytes((int)((sectorSize + sectorSkip) * length));
|
|
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
byte[] sector = new byte[sectorSize];
|
|
Array.Copy(buffer, (sectorSize + sectorSkip) * i, sector, 0, sectorSize);
|
|
sector = Sector.GetUserDataFromMode2(sector);
|
|
mode2Ms.Write(sector, 0, sector.Length);
|
|
}
|
|
|
|
buffer = mode2Ms.ToArray();
|
|
}
|
|
else if(sectorOffset == 0 && sectorSkip == 0)
|
|
buffer = br.ReadBytes((int)(sectorSize * length));
|
|
else
|
|
{
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
if(_rawDvd)
|
|
{
|
|
byte[] sector = br.ReadBytes((int)(sectorSize + sectorSkip + sectorOffset));
|
|
ErrorNumber error = _decoding.Scramble(sector, out byte[] scrambled);
|
|
|
|
if(error != ErrorNumber.NoError) return error;
|
|
|
|
Array.Copy(scrambled, sectorOffset, buffer, i * sectorSize, sectorSize);
|
|
}
|
|
else
|
|
{
|
|
br.BaseStream.Seek(sectorOffset, SeekOrigin.Current);
|
|
byte[] sector = br.ReadBytes((int)sectorSize);
|
|
br.BaseStream.Seek(sectorSkip, SeekOrigin.Current);
|
|
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ErrorNumber.NoError;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public List<Track> GetSessionTracks(Session session)
|
|
{
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return null;
|
|
|
|
if(session.Sequence != 1) return null;
|
|
|
|
var trk = new Track
|
|
{
|
|
BytesPerSector = (int)_imageInfo.SectorSize,
|
|
EndSector = _imageInfo.Sectors - 1,
|
|
Filter = _rawImageFilter,
|
|
File = _rawImageFilter.Filename,
|
|
FileOffset = 0,
|
|
FileType = "BINARY",
|
|
RawBytesPerSector = (int)_imageInfo.SectorSize,
|
|
Sequence = 1,
|
|
StartSector = 0,
|
|
SubchannelType = TrackSubchannelType.None,
|
|
Type = TrackType.Data,
|
|
Session = 1
|
|
};
|
|
|
|
List<Track> lst = [trk];
|
|
|
|
return lst;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public List<Track> GetSessionTracks(ushort session)
|
|
{
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return null;
|
|
|
|
if(session != 1) return null;
|
|
|
|
var trk = new Track
|
|
{
|
|
BytesPerSector = (int)_imageInfo.SectorSize,
|
|
EndSector = _imageInfo.Sectors - 1,
|
|
Filter = _rawImageFilter,
|
|
File = _rawImageFilter.Filename,
|
|
FileOffset = 0,
|
|
FileType = "BINARY",
|
|
RawBytesPerSector = (int)_imageInfo.SectorSize,
|
|
Sequence = 1,
|
|
StartSector = 0,
|
|
SubchannelType = TrackSubchannelType.None,
|
|
Type = TrackType.Data,
|
|
Session = 1
|
|
};
|
|
|
|
List<Track> lst = [trk];
|
|
|
|
return lst;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSector(ulong sectorAddress, uint track, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported;
|
|
|
|
return track != 1 ? ErrorNumber.OutOfRange : ReadSector(sectorAddress, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectors(ulong sectorAddress, uint length, uint track, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported;
|
|
|
|
return track != 1 ? ErrorNumber.OutOfRange : ReadSectors(sectorAddress, length, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorLong(ulong sectorAddress, uint track, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported;
|
|
|
|
return track != 1 ? ErrorNumber.OutOfRange : ReadSectorsLong(sectorAddress, 1, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, uint track, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported;
|
|
|
|
return track != 1 ? ErrorNumber.OutOfRange : ReadSectorsLong(sectorAddress, length, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorTag(ulong sectorAddress, SectorTagType tag, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc ||
|
|
!_rawCompactDisc && !_toastXa && tag != SectorTagType.CdTrackFlags)
|
|
return ErrorNumber.NotSupported;
|
|
|
|
return ReadSectorsTag(sectorAddress, 1, tag, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, SectorTagType tag, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc ||
|
|
!_rawCompactDisc && !_toastXa && !_rawDvd && tag != SectorTagType.CdTrackFlags)
|
|
return ErrorNumber.NotSupported;
|
|
|
|
if(tag == SectorTagType.CdTrackFlags)
|
|
{
|
|
buffer = [4];
|
|
|
|
return ErrorNumber.NoError;
|
|
}
|
|
|
|
if(sectorAddress > _imageInfo.Sectors - 1) return ErrorNumber.OutOfRange;
|
|
|
|
if(sectorAddress + length > _imageInfo.Sectors) return ErrorNumber.OutOfRange;
|
|
|
|
uint sectorOffset;
|
|
uint sectorSize;
|
|
uint sectorSkip = 0;
|
|
|
|
if(!_hasSubchannel && tag == SectorTagType.CdSectorSubchannel) return ErrorNumber.NoData;
|
|
|
|
// Requires reading sector
|
|
if(_mode2)
|
|
{
|
|
if(tag != SectorTagType.CdSectorSubchannel) return ErrorNumber.NotImplemented;
|
|
|
|
sectorOffset = 2352;
|
|
sectorSize = 96;
|
|
}
|
|
else if(_toastXa)
|
|
{
|
|
switch(tag)
|
|
{
|
|
case SectorTagType.CdSectorSubHeader:
|
|
sectorOffset = 0;
|
|
sectorSize = 8;
|
|
sectorSkip = 2048;
|
|
|
|
break;
|
|
|
|
default:
|
|
return ErrorNumber.NotSupported;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch(tag)
|
|
{
|
|
case SectorTagType.CdSectorSync:
|
|
{
|
|
sectorOffset = 0;
|
|
sectorSize = 12;
|
|
sectorSkip = 2340;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorHeader:
|
|
{
|
|
sectorOffset = 12;
|
|
sectorSize = 4;
|
|
sectorSkip = 2336;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorSubchannel:
|
|
{
|
|
sectorOffset = 2352;
|
|
sectorSize = 96;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorSubHeader:
|
|
return ErrorNumber.NotSupported;
|
|
case SectorTagType.CdSectorEcc:
|
|
{
|
|
sectorOffset = 2076;
|
|
sectorSize = 276;
|
|
sectorSkip = 0;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorEccP:
|
|
{
|
|
sectorOffset = 2076;
|
|
sectorSize = 172;
|
|
sectorSkip = 104;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorEccQ:
|
|
{
|
|
sectorOffset = 2248;
|
|
sectorSize = 104;
|
|
sectorSkip = 0;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.CdSectorEdc:
|
|
{
|
|
sectorOffset = 2064;
|
|
sectorSize = 4;
|
|
sectorSkip = 284;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorNumber:
|
|
{
|
|
sectorOffset = 1;
|
|
sectorSize = 3;
|
|
sectorSkip = 2060;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorInformation:
|
|
{
|
|
sectorOffset = 0;
|
|
sectorSize = 1;
|
|
sectorSkip = 2063;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorIed:
|
|
{
|
|
sectorOffset = 4;
|
|
sectorSize = 2;
|
|
sectorSkip = 2058;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorCmi:
|
|
{
|
|
sectorOffset = 6;
|
|
sectorSize = 1;
|
|
sectorSkip = 2057;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorTitleKey:
|
|
{
|
|
sectorOffset = 7;
|
|
sectorSize = 5;
|
|
sectorSkip = 2052;
|
|
|
|
break;
|
|
}
|
|
|
|
case SectorTagType.DvdSectorEdc:
|
|
{
|
|
sectorOffset = 2060;
|
|
sectorSize = 4;
|
|
sectorSkip = 0;
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
return ErrorNumber.NotSupported;
|
|
}
|
|
}
|
|
|
|
buffer = new byte[sectorSize * length];
|
|
|
|
Stream stream = _rawImageFilter.GetDataForkStream();
|
|
var br = new BinaryReader(stream);
|
|
br.BaseStream.Seek((long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip)), SeekOrigin.Begin);
|
|
|
|
if(sectorOffset == 0 && sectorSkip == 0)
|
|
buffer = br.ReadBytes((int)(sectorSize * length));
|
|
else
|
|
{
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
br.BaseStream.Seek(sectorOffset, SeekOrigin.Current);
|
|
byte[] sector = br.ReadBytes((int)sectorSize);
|
|
br.BaseStream.Seek(sectorSkip, SeekOrigin.Current);
|
|
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
|
|
}
|
|
}
|
|
|
|
return ErrorNumber.NoError;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorLong(ulong sectorAddress, out byte[] buffer) =>
|
|
ReadSectorsLong(sectorAddress, 1, out buffer);
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc || !_rawCompactDisc && !_toastXa && !_rawDvd)
|
|
return ErrorNumber.NotSupported;
|
|
|
|
if(sectorAddress > _imageInfo.Sectors - 1) return ErrorNumber.OutOfRange;
|
|
|
|
if(sectorAddress + length > _imageInfo.Sectors) return ErrorNumber.OutOfRange;
|
|
|
|
uint sectorSize = 2352u;
|
|
|
|
if(_toastXa) sectorSize = 2056u;
|
|
|
|
if(_rawDvd) sectorSize = 2064u;
|
|
|
|
uint sectorSkip = 0;
|
|
|
|
if(_hasSubchannel) sectorSkip += 96;
|
|
|
|
buffer = new byte[sectorSize * length];
|
|
|
|
Stream stream = _rawImageFilter.GetDataForkStream();
|
|
var br = new BinaryReader(stream);
|
|
|
|
br.BaseStream.Seek((long)(sectorAddress * (sectorSize + sectorSkip)), SeekOrigin.Begin);
|
|
|
|
if(_toastXa)
|
|
{
|
|
buffer = new byte[2352 * length];
|
|
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
byte[] fullSector = new byte[2352];
|
|
stream.EnsureRead(fullSector, 16, (int)sectorSize);
|
|
SectorBuilder sb = new();
|
|
sb.ReconstructPrefix(ref fullSector, TrackType.CdMode2Form1, (long)(sectorAddress + length));
|
|
sb.ReconstructEcc(ref fullSector, TrackType.CdMode2Form1);
|
|
Array.Copy(fullSector, 0, buffer, i * 2352, 2352);
|
|
}
|
|
}
|
|
else if(_rawDvd)
|
|
{
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
byte[] sector = br.ReadBytes((int)sectorSize);
|
|
ErrorNumber error = _decoding.Scramble(sector, out byte[] scrambled);
|
|
|
|
if(error != ErrorNumber.NoError) return error;
|
|
|
|
Array.Copy(scrambled, 0, buffer, i * sectorSize, sectorSize);
|
|
}
|
|
}
|
|
else if(sectorSkip == 0)
|
|
buffer = br.ReadBytes((int)(sectorSize * length));
|
|
else
|
|
{
|
|
for(int i = 0; i < length; i++)
|
|
{
|
|
byte[] sector = br.ReadBytes((int)sectorSize);
|
|
br.BaseStream.Seek(sectorSkip, SeekOrigin.Current);
|
|
|
|
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
|
|
}
|
|
}
|
|
|
|
return ErrorNumber.NoError;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadMediaTag(MediaTagType tag, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
if(!_mediaTags.TryGetValue(tag, out byte[] data)) return ErrorNumber.NoData;
|
|
|
|
buffer = data?.Clone() as byte[];
|
|
|
|
return buffer == null ? ErrorNumber.NoData : ErrorNumber.NoError;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorTag(ulong sectorAddress, uint track, SectorTagType tag, out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
return _imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc || !_rawCompactDisc
|
|
? ErrorNumber.NotSupported
|
|
: track != 1
|
|
? ErrorNumber.OutOfRange
|
|
: ReadSectorsTag(sectorAddress, 1, track, tag, out buffer);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, uint track, SectorTagType tag,
|
|
out byte[] buffer)
|
|
{
|
|
buffer = null;
|
|
|
|
return _imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc || !_rawCompactDisc
|
|
? ErrorNumber.NotSupported
|
|
: track != 1
|
|
? ErrorNumber.OutOfRange
|
|
: ReadSectorsTag(sectorAddress, length, tag, out buffer);
|
|
}
|
|
|
|
#endregion
|
|
} |