// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : Read.cs // Author(s) : Natalia Portillo // // Component : Disk image plugins. // // --[ Description ] ---------------------------------------------------------- // // Reads Aaru Format disk 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 . // // ---------------------------------------------------------------------------- // Copyright © 2011-2025 Natalia Portillo // Copyright © 2020-2025 Rebecca Wallander // ****************************************************************************/ using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Linq; using System.Runtime.InteropServices; using System.Text; using System.Text.Json; using System.Xml; using System.Xml.Serialization; using Aaru.Checksums; using Aaru.CommonTypes; using Aaru.CommonTypes.AaruMetadata; using Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Compression; using Aaru.Decoders.CD; using Aaru.Helpers; using Aaru.Logging; using Humanizer; using Schemas; using Marshal = Aaru.Helpers.Marshal; using Partition = Aaru.CommonTypes.Partition; using Session = Aaru.CommonTypes.Structs.Session; using TapeFile = Aaru.CommonTypes.Structs.TapeFile; using TapePartition = Aaru.CommonTypes.Structs.TapePartition; using Track = Aaru.CommonTypes.Structs.Track; using TrackType = Aaru.CommonTypes.Enums.TrackType; namespace Aaru.Images; public sealed partial class AaruFormat { #region IWritableOpticalImage Members /// public ErrorNumber Open(IFilter imageFilter) { AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); _imageStream = imageFilter.GetDataForkStream(); _imageStream.Seek(0, SeekOrigin.Begin); if(_imageStream.Length < Marshal.SizeOf()) return ErrorNumber.InvalidArgument; _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); _header = Marshal.ByteArrayToStructureLittleEndian(_structureBytes); if(_header.imageMajorVersion > AARUFMT_VERSION) { AaruLogging.Error(string.Format(Localization.Image_version_0_not_recognized, _header.imageMajorVersion)); return ErrorNumber.NotSupported; } _imageInfo.Application = _header.application; _imageInfo.ApplicationVersion = $"{_header.applicationMajorVersion}.{_header.applicationMinorVersion}"; _imageInfo.Version = $"{_header.imageMajorVersion}.{_header.imageMinorVersion}"; _imageInfo.MediaType = _header.mediaType; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Read the index header _imageStream.Position = (long)_header.indexOffset; _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); IndexHeader idxHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); if(idxHeader.identifier != BlockType.Index && idxHeader.identifier != BlockType.Index2) { AaruLogging.Error(Localization.Index_not_found); return ErrorNumber.InvalidArgument; } if(idxHeader.identifier == BlockType.Index2) { _imageStream.Position = (long)_header.indexOffset; _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); IndexHeader2 idxHeader2 = Marshal.SpanToStructureLittleEndian(_structureBytes); AaruLogging.Debug(MODULE_NAME, Localization.Index_at_0_contains_1_entries, _header.indexOffset, idxHeader2.entries); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Fill in-memory index _index = []; for(ulong i = 0; i < idxHeader2.entries; i++) { _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); IndexEntry entry = Marshal.SpanToStructureLittleEndian(_structureBytes); AaruLogging.Debug(MODULE_NAME, Localization.Block_type_0_with_data_type_1_is_indexed_to_be_at_2, entry.blockType, entry.dataType, entry.offset); _index.Add(entry); } } else { AaruLogging.Debug(MODULE_NAME, Localization.Index_at_0_contains_1_entries, _header.indexOffset, idxHeader.entries); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Fill in-memory index _index = []; for(ushort i = 0; i < idxHeader.entries; i++) { _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); IndexEntry entry = Marshal.SpanToStructureLittleEndian(_structureBytes); AaruLogging.Debug(MODULE_NAME, Localization.Block_type_0_with_data_type_1_is_indexed_to_be_at_2, entry.blockType, entry.dataType, entry.offset); _index.Add(entry); } } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); _imageInfo.ImageSize = 0; bool foundUserDataDdt = false; _mediaTags = new Dictionary(); List compactDiscIndexes = null; foreach(IndexEntry entry in _index) { _imageStream.Position = (long)entry.offset; switch(entry.blockType) { case BlockType.DataBlock: // NOP block, skip if(entry.dataType == DataType.NoData) break; _imageStream.Position = (long)entry.offset; _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); BlockHeader blockHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); _imageInfo.ImageSize += blockHeader.cmpLength; // Unused, skip if(entry.dataType == DataType.UserData) { if(blockHeader.sectorSize > _imageInfo.SectorSize) _imageInfo.SectorSize = blockHeader.sectorSize; break; } if(blockHeader.identifier != entry.blockType) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_identifier_for_data_block_at_position_0, entry.offset); break; } if(blockHeader.type != entry.dataType) { AaruLogging.Debug(MODULE_NAME, Localization .Expected_block_with_data_type_0_at_position_1_but_found_data_type_2, entry.dataType, entry.offset, blockHeader.type); break; } byte[] data; AaruLogging.Debug(MODULE_NAME, Localization.Found_data_block_type_0_at_position_1, entry.dataType, entry.offset); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Decompress media tag if(blockHeader.compression is CompressionType.Lzma or CompressionType.LzmaClauniaSubchannelTransform) { if(blockHeader.compression == CompressionType.LzmaClauniaSubchannelTransform && entry.dataType != DataType.CdSectorSubchannel) { AaruLogging.Debug(MODULE_NAME, Localization .Invalid_compression_type_0_for_block_with_data_type_1_continuing, blockHeader.compression, entry.dataType); break; } var decompressStopwatch = new Stopwatch(); decompressStopwatch.Start(); byte[] compressedTag = new byte[blockHeader.cmpLength - LZMA_PROPERTIES_LENGTH]; byte[] lzmaProperties = new byte[LZMA_PROPERTIES_LENGTH]; _imageStream.EnsureRead(lzmaProperties, 0, LZMA_PROPERTIES_LENGTH); _imageStream.EnsureRead(compressedTag, 0, compressedTag.Length); data = new byte[blockHeader.length]; int decompressedLength = LZMA.DecodeBuffer(compressedTag, data, lzmaProperties); if(decompressedLength != blockHeader.length) { AaruLogging.Debug(MODULE_NAME, Localization.Error_decompressing_block_should_be_0_bytes_but_got_1_bytes, blockHeader.length, decompressedLength); return ErrorNumber.InOutError; } if(blockHeader.compression == CompressionType.LzmaClauniaSubchannelTransform) data = ClauniaSubchannelUntransform(data); decompressStopwatch.Stop(); AaruLogging.Debug(MODULE_NAME, Localization.Took_0_seconds_to_decompress_block, decompressStopwatch.Elapsed.TotalSeconds); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); } else if(blockHeader.compression == CompressionType.None) { data = new byte[blockHeader.length]; _imageStream.EnsureRead(data, 0, (int)blockHeader.length); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); } else { AaruLogging.Debug(MODULE_NAME, Localization.Found_unknown_compression_type_0_continuing, (ushort)blockHeader.compression); break; } // Check CRC, if not correct, skip it Crc64Context.Data(data, out byte[] blockCrc); if(BitConverter.ToUInt64(blockCrc, 0) != blockHeader.crc64 && blockHeader.crc64 != 0) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_CRC_found_0_X16_found_expected_1_X16_continuing, BitConverter.ToUInt64(blockCrc, 0), blockHeader.crc64); break; } // Check if it's not a media tag, but a sector tag, and fill the appropriate table then switch(entry.dataType) { case DataType.CdSectorPrefix: case DataType.CdSectorPrefixCorrected: if(entry.dataType == DataType.CdSectorPrefixCorrected) { _sectorPrefixMs ??= new MemoryStream(); _sectorPrefixMs.Write(data, 0, data.Length); } else _sectorPrefix = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync)) _imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync); if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader)) _imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.CdSectorSuffix: case DataType.CdSectorSuffixCorrected: if(entry.dataType == DataType.CdSectorSuffixCorrected) { _sectorSuffixMs ??= new MemoryStream(); _sectorSuffixMs.Write(data, 0, data.Length); } else _sectorSuffix = data; 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); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.CdSectorSubchannel: _sectorSubchannel = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubchannel)) _imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubchannel); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.AppleProfileTag: case DataType.AppleSonyTag: case DataType.PriamDataTowerTag: _sectorSubchannel = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.AppleSectorTag)) _imageInfo.ReadableSectorTags.Add(SectorTagType.AppleSectorTag); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.CompactDiscMode2Subheader: _mode2Subheaders = data; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.DvdSectorCprMai: _sectorCprMai = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorCmi)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorCmi); if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorTitleKey)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorTitleKey); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.DvdSectorId: _sectorId = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorInformation)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorInformation); if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorNumber)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorNumber); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.DvdSectorIed: _sectorIed = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorIed)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorIed); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.DvdSectorEdc: _sectorEdc = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdSectorEdc)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdSectorEdc); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case DataType.DvdSectorTitleKeyDecrypted: _sectorDecryptedTitleKey = data; if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.DvdTitleKeyDecrypted)) _imageInfo.ReadableSectorTags.Add(SectorTagType.DvdTitleKeyDecrypted); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; default: MediaTagType mediaTagType = GetMediaTagTypeForDataType(blockHeader.type); if(_mediaTags.ContainsKey(mediaTagType)) { AaruLogging.Debug(MODULE_NAME, Localization.Media_tag_type_0_duplicated_removing_previous_entry, mediaTagType); _mediaTags.Remove(mediaTagType); } _mediaTags.Add(mediaTagType, data); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; } break; case BlockType.DeDuplicationTable: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); DdtHeader ddtHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); _imageInfo.ImageSize += ddtHeader.cmpLength; if(ddtHeader.identifier != BlockType.DeDuplicationTable) break; switch(entry.dataType) { case DataType.UserData: _imageInfo.Sectors = ddtHeader.entries; _shift = ddtHeader.shift; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Check for DDT compression switch(ddtHeader.compression) { case CompressionType.Lzma: AaruLogging.Debug(MODULE_NAME, Localization.Decompressing_DDT); var ddtStopwatch = new Stopwatch(); ddtStopwatch.Start(); byte[] compressedDdt = new byte[ddtHeader.cmpLength - LZMA_PROPERTIES_LENGTH]; byte[] lzmaProperties = new byte[LZMA_PROPERTIES_LENGTH]; _imageStream.EnsureRead(lzmaProperties, 0, LZMA_PROPERTIES_LENGTH); _imageStream.EnsureRead(compressedDdt, 0, compressedDdt.Length); byte[] decompressedDdt = new byte[ddtHeader.length]; ulong decompressedLength = (ulong)LZMA.DecodeBuffer(compressedDdt, decompressedDdt, lzmaProperties); if(decompressedLength != ddtHeader.length) { AaruLogging.Debug(MODULE_NAME, Localization .Error_decompressing_DDT_should_be_0_bytes_but_got_1_bytes, ddtHeader.length, decompressedLength); return ErrorNumber.InOutError; } _userDataDdt = MemoryMarshal.Cast(decompressedDdt).ToArray(); ddtStopwatch.Stop(); _inMemoryDdt = true; AaruLogging.Debug(MODULE_NAME, Localization.Took_0_seconds_to_decompress_DDT, ddtStopwatch.Elapsed.TotalSeconds); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case CompressionType.None: _inMemoryDdt = false; _outMemoryDdtPosition = (long)entry.offset; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; default: AaruLogging.Error(string.Format(Localization .Found_unsupported_compression_algorithm_0, (ushort)ddtHeader.compression)); return ErrorNumber.NotSupported; } foundUserDataDdt = true; break; case DataType.CdSectorPrefixCorrected: case DataType.CdSectorSuffixCorrected: { byte[] decompressedDdt = new byte[ddtHeader.length]; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Check for DDT compression switch(ddtHeader.compression) { case CompressionType.Lzma: AaruLogging.Debug(MODULE_NAME, Localization.Decompressing_DDT); var ddtStopwatch = new Stopwatch(); ddtStopwatch.Start(); byte[] compressedDdt = new byte[ddtHeader.cmpLength - LZMA_PROPERTIES_LENGTH]; byte[] lzmaProperties = new byte[LZMA_PROPERTIES_LENGTH]; _imageStream.EnsureRead(lzmaProperties, 0, LZMA_PROPERTIES_LENGTH); _imageStream.EnsureRead(compressedDdt, 0, compressedDdt.Length); ulong decompressedLength = (ulong)LZMA.DecodeBuffer(compressedDdt, decompressedDdt, lzmaProperties); ddtStopwatch.Stop(); if(decompressedLength != ddtHeader.length) { AaruLogging.Debug(MODULE_NAME, Localization .Error_decompressing_DDT_should_be_0_bytes_but_got_1_bytes, ddtHeader.length, decompressedLength); return ErrorNumber.InOutError; } AaruLogging.Debug(MODULE_NAME, Localization.Took_0_seconds_to_decompress_DDT, ddtStopwatch.Elapsed.TotalSeconds); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case CompressionType.None: _imageStream.EnsureRead(decompressedDdt, 0, decompressedDdt.Length); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; default: AaruLogging.Error(string.Format(Localization .Found_unsupported_compression_algorithm_0, (ushort)ddtHeader.compression)); return ErrorNumber.NotSupported; } uint[] cdDdt = MemoryMarshal.Cast(decompressedDdt).ToArray(); switch(entry.dataType) { case DataType.CdSectorPrefixCorrected: _sectorPrefixDdt = cdDdt; _sectorPrefixMs = new MemoryStream(); break; case DataType.CdSectorSuffixCorrected: _sectorSuffixDdt = cdDdt; _sectorSuffixMs = new MemoryStream(); break; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; } } break; // Logical geometry block. It doesn't have a CRC coz, well, it's not so important case BlockType.GeometryBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); _geometryBlock = Marshal.SpanToStructureLittleEndian(_structureBytes); if(_geometryBlock.identifier == BlockType.GeometryBlock) { AaruLogging.Debug(MODULE_NAME, Localization.Geometry_set_to_0_cylinders_1_heads_2_sectors_per_track, _geometryBlock.cylinders, _geometryBlock.heads, _geometryBlock.sectorsPerTrack); _imageInfo.Cylinders = _geometryBlock.cylinders; _imageInfo.Heads = _geometryBlock.heads; _imageInfo.SectorsPerTrack = _geometryBlock.sectorsPerTrack; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); } break; // Metadata block case BlockType.MetadataBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); MetadataBlock metadataBlock = Marshal.SpanToStructureLittleEndian(_structureBytes); if(metadataBlock.identifier != entry.blockType) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_identifier_for_data_block_at_position_0, entry.offset); break; } AaruLogging.Debug(MODULE_NAME, Localization.Found_metadata_block_at_position_0, entry.offset); byte[] metadata = new byte[metadataBlock.blockSize]; _imageStream.Position = (long)entry.offset; _imageStream.EnsureRead(metadata, 0, metadata.Length); if(metadataBlock is { mediaSequence: > 0, lastMediaSequence: > 0 }) { _imageInfo.MediaSequence = metadataBlock.mediaSequence; _imageInfo.LastMediaSequence = metadataBlock.lastMediaSequence; AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_sequence_as_0_of_1, _imageInfo.MediaSequence, _imageInfo.LastMediaSequence); } if(metadataBlock.creatorLength > 0 && metadataBlock.creatorLength + metadataBlock.creatorOffset <= metadata.Length) { _imageInfo.Creator = Encoding.Unicode.GetString(metadata, (int)metadataBlock.creatorOffset, (int)(metadataBlock.creatorLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_creator_0, _imageInfo.Creator); } if(metadataBlock.commentsOffset > 0 && metadataBlock.commentsLength + metadataBlock.commentsOffset <= metadata.Length) { _imageInfo.Comments = Encoding.Unicode.GetString(metadata, (int)metadataBlock.commentsOffset, (int)(metadataBlock.commentsLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_comments_0, _imageInfo.Comments); } if(metadataBlock.mediaTitleOffset > 0 && metadataBlock.mediaTitleLength + metadataBlock.mediaTitleOffset <= metadata.Length) { _imageInfo.MediaTitle = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaTitleOffset, (int)(metadataBlock.mediaTitleLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_title_0, _imageInfo.MediaTitle); } if(metadataBlock.mediaManufacturerOffset > 0 && metadataBlock.mediaManufacturerLength + metadataBlock.mediaManufacturerOffset <= metadata.Length) { _imageInfo.MediaManufacturer = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaManufacturerOffset, (int)(metadataBlock.mediaManufacturerLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_manufacturer_0, _imageInfo.MediaManufacturer); } if(metadataBlock.mediaModelOffset > 0 && metadataBlock.mediaModelLength + metadataBlock.mediaModelOffset <= metadata.Length) { _imageInfo.MediaModel = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaModelOffset, (int)(metadataBlock.mediaModelLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_model_0, _imageInfo.MediaModel); } if(metadataBlock.mediaSerialNumberOffset > 0 && metadataBlock.mediaSerialNumberLength + metadataBlock.mediaSerialNumberOffset <= metadata.Length) { _imageInfo.MediaSerialNumber = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaSerialNumberOffset, (int)(metadataBlock.mediaSerialNumberLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_serial_number_0, _imageInfo.MediaSerialNumber); } if(metadataBlock.mediaBarcodeOffset > 0 && metadataBlock.mediaBarcodeLength + metadataBlock.mediaBarcodeOffset <= metadata.Length) { _imageInfo.MediaBarcode = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaBarcodeOffset, (int)(metadataBlock.mediaBarcodeLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_barcode_0, _imageInfo.MediaBarcode); } if(metadataBlock.mediaPartNumberOffset > 0 && metadataBlock.mediaPartNumberLength + metadataBlock.mediaPartNumberOffset <= metadata.Length) { _imageInfo.MediaPartNumber = Encoding.Unicode.GetString(metadata, (int)metadataBlock.mediaPartNumberOffset, (int)(metadataBlock.mediaPartNumberLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_media_part_number_0, _imageInfo.MediaPartNumber); } if(metadataBlock.driveManufacturerOffset > 0 && metadataBlock.driveManufacturerLength + metadataBlock.driveManufacturerOffset <= metadata.Length) { _imageInfo.DriveManufacturer = Encoding.Unicode.GetString(metadata, (int)metadataBlock.driveManufacturerOffset, (int)(metadataBlock.driveManufacturerLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_drive_manufacturer_0, _imageInfo.DriveManufacturer); } if(metadataBlock.driveModelOffset > 0 && metadataBlock.driveModelLength + metadataBlock.driveModelOffset <= metadata.Length) { _imageInfo.DriveModel = Encoding.Unicode.GetString(metadata, (int)metadataBlock.driveModelOffset, (int)(metadataBlock.driveModelLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_drive_model_0, _imageInfo.DriveModel); } if(metadataBlock.driveSerialNumberOffset > 0 && metadataBlock.driveSerialNumberLength + metadataBlock.driveSerialNumberOffset <= metadata.Length) { _imageInfo.DriveSerialNumber = Encoding.Unicode.GetString(metadata, (int)metadataBlock.driveSerialNumberOffset, (int)(metadataBlock.driveSerialNumberLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_drive_serial_number_0, _imageInfo.DriveSerialNumber); } if(metadataBlock.driveFirmwareRevisionOffset > 0 && metadataBlock.driveFirmwareRevisionLength + metadataBlock.driveFirmwareRevisionOffset <= metadata.Length) { _imageInfo.DriveFirmwareRevision = Encoding.Unicode.GetString(metadata, (int)metadataBlock.driveFirmwareRevisionOffset, (int)(metadataBlock.driveFirmwareRevisionLength - 2)); AaruLogging.Debug(MODULE_NAME, Localization.Setting_drive_firmware_revision_0, _imageInfo.DriveFirmwareRevision); } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; // Optical disc tracks block case BlockType.TracksBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); TracksHeader tracksHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); if(tracksHeader.identifier != BlockType.TracksBlock) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_identifier_for_tracks_block_at_position_0, entry.offset); break; } _structureBytes = new byte[Marshal.SizeOf() * tracksHeader.entries]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); Crc64Context.Data(_structureBytes, out byte[] trksCrc); if(BitConverter.ToUInt64(trksCrc, 0) != tracksHeader.crc64) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_CRC_found_0_X16_found_expected_1_X16_continuing, BitConverter.ToUInt64(trksCrc, 0), tracksHeader.crc64); break; } _imageStream.Position -= _structureBytes.Length; Tracks = []; _trackFlags = new Dictionary(); _trackIsrcs = new Dictionary(); AaruLogging.Debug(MODULE_NAME, Localization.Found_0_tracks_at_position_1, tracksHeader.entries, entry.offset); for(ushort i = 0; i < tracksHeader.entries; i++) { _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); TrackEntry trackEntry = Marshal.ByteArrayToStructureLittleEndian(_structureBytes); Tracks.Add(new Track { Sequence = trackEntry.sequence, Type = trackEntry.type, StartSector = (ulong)trackEntry.start, EndSector = (ulong)trackEntry.end, Pregap = (ulong)trackEntry.pregap, Session = trackEntry.session, File = imageFilter.Filename, FileType = "BINARY", Filter = imageFilter }); if(trackEntry.type == TrackType.Data) continue; _trackFlags.Add(trackEntry.sequence, trackEntry.flags); if(!string.IsNullOrEmpty(trackEntry.isrc)) _trackIsrcs.Add(trackEntry.sequence, trackEntry.isrc); } if(_trackFlags.Count > 0 && !_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdTrackFlags)) _imageInfo.ReadableSectorTags.Add(SectorTagType.CdTrackFlags); if(_trackIsrcs.Count > 0 && !_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdTrackIsrc)) _imageInfo.ReadableSectorTags.Add(SectorTagType.CdTrackIsrc); _imageInfo.HasPartitions = true; _imageInfo.HasSessions = true; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; // CICM XML metadata block case BlockType.CicmBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); CicmMetadataBlock cicmBlock = Marshal.SpanToStructureLittleEndian(_structureBytes); if(cicmBlock.identifier != BlockType.CicmBlock) break; AaruLogging.Debug(MODULE_NAME, Localization.Found_CICM_XML_metadata_block_at_position_0, entry.offset); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); byte[] cicmBytes = new byte[cicmBlock.length]; _imageStream.EnsureRead(cicmBytes, 0, cicmBytes.Length); var cicmMs = new MemoryStream(cicmBytes); // The converter to AaruMetadata basically overcomes this (should?) #pragma warning disable IL2026 var cicmXs = new XmlSerializer(typeof(CICMMetadataType)); #pragma warning restore IL2026 try { var sr = new StreamReader(cicmMs); // The converter to AaruMetadata basically overcomes this (should?) #pragma warning disable IL2026 AaruMetadata = (CICMMetadataType)cicmXs.Deserialize(sr); #pragma warning restore IL2026 sr.Close(); } catch(XmlException ex) { AaruLogging.Debug(MODULE_NAME, Localization.Exception_0_processing_CICM_XML_metadata_block, ex.Message); AaruLogging.Exception(ex, Localization.Exception_0_processing_CICM_XML_metadata_block, ex.Message); AaruMetadata = null; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; // Aaru Metadata JSON block case BlockType.AaruMetadataJsonBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); AaruMetadataJsonBlock aaruMetadataBlock = Marshal.SpanToStructureLittleEndian(_structureBytes); if(aaruMetadataBlock.identifier != BlockType.AaruMetadataJsonBlock) break; AaruLogging.Debug(MODULE_NAME, Localization.Found_Aaru_Metadata_block_at_position_0, entry.offset); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); byte[] jsonBytes = new byte[aaruMetadataBlock.length]; _imageStream.EnsureRead(jsonBytes, 0, jsonBytes.Length); try { AaruMetadata = (JsonSerializer.Deserialize(jsonBytes, typeof(MetadataJson), MetadataJsonContext.Default) as MetadataJson)?.AaruMetadata; } catch(JsonException ex) { AaruLogging.Debug(MODULE_NAME, Localization.Exception_0_processing_Aaru_Metadata_block, ex.Message); AaruLogging.Exception(ex, Localization.Exception_0_processing_Aaru_Metadata_block, ex.Message); AaruMetadata = null; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; // Dump hardware block case BlockType.DumpHardwareBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); DumpHardwareHeader dumpBlock = Marshal.SpanToStructureLittleEndian(_structureBytes); if(dumpBlock.identifier != BlockType.DumpHardwareBlock) break; AaruLogging.Debug(MODULE_NAME, Localization.Found_dump_hardware_block_at_position_0, entry.offset); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); _structureBytes = new byte[dumpBlock.length]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); Crc64Context.Data(_structureBytes, out byte[] dumpCrc); if(BitConverter.ToUInt64(dumpCrc, 0) != dumpBlock.crc64) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_CRC_found_0_X16_found_expected_1_X16_continuing, BitConverter.ToUInt64(dumpCrc, 0), dumpBlock.crc64); break; } _imageStream.Position -= _structureBytes.Length; DumpHardware = []; for(ushort i = 0; i < dumpBlock.entries; i++) { _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); DumpHardwareEntry dumpEntry = Marshal.SpanToStructureLittleEndian(_structureBytes); var dump = new DumpHardware { Software = new Software(), Extents = [] }; byte[] tmp; if(dumpEntry.manufacturerLength > 0) { tmp = new byte[dumpEntry.manufacturerLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Manufacturer = Encoding.UTF8.GetString(tmp); } if(dumpEntry.modelLength > 0) { tmp = new byte[dumpEntry.modelLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Model = Encoding.UTF8.GetString(tmp); } if(dumpEntry.revisionLength > 0) { tmp = new byte[dumpEntry.revisionLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Revision = Encoding.UTF8.GetString(tmp); } if(dumpEntry.firmwareLength > 0) { tmp = new byte[dumpEntry.firmwareLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Firmware = Encoding.UTF8.GetString(tmp); } if(dumpEntry.serialLength > 0) { tmp = new byte[dumpEntry.serialLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Serial = Encoding.UTF8.GetString(tmp); } if(dumpEntry.softwareNameLength > 0) { tmp = new byte[dumpEntry.softwareNameLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Software.Name = Encoding.UTF8.GetString(tmp); } if(dumpEntry.softwareVersionLength > 0) { tmp = new byte[dumpEntry.softwareVersionLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Software.Version = Encoding.UTF8.GetString(tmp); } if(dumpEntry.softwareOperatingSystemLength > 0) { tmp = new byte[dumpEntry.softwareOperatingSystemLength - 1]; _imageStream.EnsureRead(tmp, 0, tmp.Length); _imageStream.Position += 1; dump.Software.OperatingSystem = Encoding.UTF8.GetString(tmp); } tmp = new byte[16]; for(uint j = 0; j < dumpEntry.extents; j++) { _imageStream.EnsureRead(tmp, 0, tmp.Length); dump.Extents.Add(new Extent { Start = BitConverter.ToUInt64(tmp, 0), End = BitConverter.ToUInt64(tmp, 8) }); } dump.Extents = dump.Extents.OrderBy(t => t.Start).ToList(); if(dump.Extents.Count > 0) DumpHardware.Add(dump); } if(DumpHardware.Count == 0) DumpHardware = null; AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; // Tape partition block case BlockType.TapePartitionBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); TapePartitionHeader partitionHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); if(partitionHeader.identifier != BlockType.TapePartitionBlock) break; AaruLogging.Debug(MODULE_NAME, Localization.Found_tape_partition_block_at_position_0, entry.offset); byte[] tapePartitionBytes = new byte[partitionHeader.length]; _imageStream.EnsureRead(tapePartitionBytes, 0, tapePartitionBytes.Length); ReadOnlySpan tapePartitions = MemoryMarshal.Cast(tapePartitionBytes); TapePartitions = []; foreach(TapePartitionEntry tapePartition in tapePartitions) { TapePartitions.Add(new TapePartition { FirstBlock = tapePartition.FirstBlock, LastBlock = tapePartition.LastBlock, Number = tapePartition.Number }); } IsTape = true; break; // Tape file block case BlockType.TapeFileBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); TapeFileHeader fileHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); if(fileHeader.identifier != BlockType.TapeFileBlock) break; AaruLogging.Debug(MODULE_NAME, Localization.Found_tape_file_block_at_position_0, entry.offset); byte[] tapeFileBytes = new byte[fileHeader.length]; _imageStream.EnsureRead(tapeFileBytes, 0, tapeFileBytes.Length); ReadOnlySpan tapeFiles = MemoryMarshal.Cast(tapeFileBytes); Files = []; foreach(TapeFileEntry file in tapeFiles) { Files.Add(new TapeFile { FirstBlock = file.FirstBlock, LastBlock = file.LastBlock, Partition = file.Partition, File = file.File }); } IsTape = true; break; // Optical disc tracks block case BlockType.CompactDiscIndexesBlock: _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); CompactDiscIndexesHeader indexesHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); if(indexesHeader.identifier != BlockType.CompactDiscIndexesBlock) { AaruLogging.Debug(MODULE_NAME, Localization .Incorrect_identifier_for_compact_disc_indexes_block_at_position_0, entry.offset); break; } _structureBytes = new byte[Marshal.SizeOf() * indexesHeader.entries]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); Crc64Context.Data(_structureBytes, out byte[] idsxCrc); if(BitConverter.ToUInt64(idsxCrc, 0) != indexesHeader.crc64) { AaruLogging.Debug(MODULE_NAME, Localization.Incorrect_CRC_found_0_X16_found_expected_1_X16_continuing, BitConverter.ToUInt64(idsxCrc, 0), indexesHeader.crc64); break; } _imageStream.Position -= _structureBytes.Length; compactDiscIndexes = []; AaruLogging.Debug(MODULE_NAME, Localization.Found_0_compact_disc_indexes_at_position_1, indexesHeader.entries, entry.offset); for(ushort i = 0; i < indexesHeader.entries; i++) { _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); compactDiscIndexes.Add(Marshal .ByteArrayToStructureLittleEndian< CompactDiscIndexEntry>(_structureBytes)); } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; } } if(!foundUserDataDdt) { AaruLogging.Error(Localization.Could_not_find_user_data_deduplication_table); return ErrorNumber.InvalidArgument; } _imageInfo.CreationTime = DateTime.FromFileTimeUtc(_header.creationTime); AaruLogging.Debug(MODULE_NAME, Localization.Image_created_on_0, _imageInfo.CreationTime); _imageInfo.LastModificationTime = DateTime.FromFileTimeUtc(_header.lastWrittenTime); AaruLogging.Debug(MODULE_NAME, Localization.Image_last_written_on_0, _imageInfo.LastModificationTime); _imageInfo.MetadataMediaType = GetMetadataMediaType(_header.mediaType); if(_geometryBlock.identifier != BlockType.GeometryBlock && _imageInfo.MetadataMediaType == MetadataMediaType.BlockMedia) { _imageInfo.Cylinders = (uint)(_imageInfo.Sectors / 16 / 63); _imageInfo.Heads = 16; _imageInfo.SectorsPerTrack = 63; } _imageInfo.ReadableMediaTags.AddRange(_mediaTags.Keys); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Initialize caches _blockCache = new Dictionary(); _blockHeaderCache = new Dictionary(); _currentCacheSize = 0; if(!_inMemoryDdt) _ddtEntryCache = new Dictionary(); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); // Initialize tracks, sessions and partitions if(_imageInfo.MetadataMediaType == MetadataMediaType.OpticalDisc) { if(Tracks != null) { bool leadOutFixed = false; bool sessionPregapFixed = false; if(_mediaTags.TryGetValue(MediaTagType.CD_FullTOC, out byte[] fullToc)) { byte[] tmp = new byte[fullToc.Length + 2]; Array.Copy(fullToc, 0, tmp, 2, fullToc.Length); tmp[0] = (byte)(fullToc.Length >> 8); tmp[1] = (byte)(fullToc.Length & 0xFF); FullTOC.CDFullTOC? decodedFullToc = FullTOC.Decode(tmp); if(decodedFullToc.HasValue) { Dictionary leadOutStarts = new(); // Lead-out starts foreach(FullTOC.TrackDataDescriptor trk in decodedFullToc.Value.TrackDescriptors.Where(trk => trk.ADR is 1 or 4 && trk.POINT == 0xA2)) { int phour, pmin, psec, pframe; if(trk.PFRAME == 0) { pframe = 74; if(trk.PSEC == 0) { psec = 59; if(trk.PMIN == 0) { pmin = 59; phour = trk.PHOUR - 1; } else { pmin = trk.PMIN - 1; phour = trk.PHOUR; } } else { psec = trk.PSEC - 1; pmin = trk.PMIN; phour = trk.PHOUR; } } else { pframe = trk.PFRAME - 1; psec = trk.PSEC; pmin = trk.PMIN; phour = trk.PHOUR; } int lastSector = phour * 3600 * 75 + pmin * 60 * 75 + psec * 75 + pframe - 150; leadOutStarts?.Add(trk.SessionNumber, lastSector + 1); } foreach(KeyValuePair leadOuts in leadOutStarts) { var lastTrackInSession = new Track(); foreach(Track trk in Tracks.Where(trk => trk.Session == leadOuts.Key) .Where(trk => trk.Sequence > lastTrackInSession.Sequence)) lastTrackInSession = trk; if(lastTrackInSession.Sequence == 0 || lastTrackInSession.EndSector == (ulong)leadOuts.Value - 1) continue; lastTrackInSession.EndSector = (ulong)leadOuts.Value - 1; leadOutFixed = true; } } } if(_header.imageMajorVersion <= 1) { foreach(Track track in Tracks) { if(track.Sequence <= 1) continue; uint firstTrackNumberInSameSession = Tracks.Where(t => t.Session == track.Session).Min(t => t.Sequence); if(firstTrackNumberInSameSession != track.Sequence) continue; if(track.Pregap == 150) continue; long dif = (long)track.Pregap - 150; track.Pregap = (ulong)((long)track.Pregap - dif); track.StartSector = (ulong)((long)track.StartSector + dif); sessionPregapFixed = true; } } if(leadOutFixed) AaruLogging.Error(Localization.This_image_has_a_corrupted_track_list_convert_will_fix_it); if(sessionPregapFixed) { AaruLogging.Error(Localization .This_image_has_a_corrupted_track_list_a_best_effort_has_been_tried_but_may_require_manual_editing_or_redump); } } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); if(Tracks == null || Tracks.Count == 0) { Tracks = [ new Track { BytesPerSector = (int)_imageInfo.SectorSize, EndSector = _imageInfo.Sectors - 1, File = imageFilter.Filename, FileType = "BINARY", Filter = imageFilter, RawBytesPerSector = (int)_imageInfo.SectorSize, Session = 1, Sequence = 1, Type = TrackType.Data } ]; _trackFlags = new Dictionary { { 1, (byte)CdFlags.DataTrack } }; _trackIsrcs = new Dictionary(); } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); Sessions = []; for(int i = 1; i <= Tracks.Max(t => t.Session); i++) { Sessions.Add(new Session { Sequence = (ushort)i, StartTrack = Tracks.Where(t => t.Session == i).Min(t => t.Sequence), EndTrack = Tracks.Where(t => t.Session == i).Max(t => t.Sequence), StartSector = Tracks.Where(t => t.Session == i).Min(t => t.StartSector), EndSector = Tracks.Where(t => t.Session == i).Max(t => t.EndSector) }); } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); foreach(Track track in Tracks.OrderBy(t => t.StartSector)) { switch(track.Sequence) { case 0: track.Pregap = 150; track.Indexes[0] = -150; track.Indexes[1] = (int)track.StartSector; continue; case 1 when Tracks.All(t => t.Sequence != 0): track.Pregap = 150; track.Indexes[0] = -150; track.Indexes[1] = (int)track.StartSector; continue; } if(track.Pregap > 0) { track.Indexes[0] = (int)track.StartSector; track.Indexes[1] = (int)(track.StartSector + track.Pregap); } else track.Indexes[1] = (int)track.StartSector; } ulong currentTrackOffset = 0; Partitions = []; foreach(Track track in Tracks.OrderBy(t => t.StartSector)) { Partitions.Add(new Partition { Sequence = track.Sequence, Type = track.Type.Humanize(), Name = string.Format(Localization.Track_0, track.Sequence), Offset = currentTrackOffset, Start = (ulong)track.Indexes[1], Size = (track.EndSector - (ulong)track.Indexes[1] + 1) * (ulong)track.BytesPerSector, Length = track.EndSector - (ulong)track.Indexes[1] + 1, Scheme = Localization.Optical_disc_track }); currentTrackOffset += (track.EndSector - track.StartSector + 1) * (ulong)track.BytesPerSector; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); Track[] tracks = Tracks.ToArray(); foreach(Track trk in tracks) { ErrorNumber errno = ReadSector(trk.StartSector, out byte[] sector); if(errno != ErrorNumber.NoError) continue; trk.BytesPerSector = sector.Length; trk.RawBytesPerSector = _sectorPrefix != null && _sectorSuffix != null || _sectorPrefixDdt != null && _sectorSuffixDdt != null ? 2352 : sector.Length; if(_sectorSubchannel == null) continue; trk.SubchannelFile = trk.File; trk.SubchannelFilter = trk.Filter; trk.SubchannelType = TrackSubchannelType.Raw; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); Tracks = tracks.ToList(); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); if(compactDiscIndexes != null) { foreach(CompactDiscIndexEntry compactDiscIndex in compactDiscIndexes.OrderBy(i => i.Track) .ThenBy(i => i.Index)) { Track track = Tracks.FirstOrDefault(t => t.Sequence == compactDiscIndex.Track); if(track is null) continue; track.Indexes[compactDiscIndex.Index] = compactDiscIndex.Lba; } } } else { Tracks = null; Sessions = null; Partitions = null; } SetMetadataFromTags(); if(_sectorSuffixDdt != null) EccInit(); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NoError; if(_imageInfo.MediaType is MediaType.CD or MediaType.CDDA or MediaType.CDG or MediaType.CDEG or MediaType.CDI or MediaType.CDROM or MediaType.CDROMXA or MediaType.CDPLUS or MediaType.CDMO or MediaType.CDR or MediaType.CDRW or MediaType.CDMRW or MediaType.VCD or MediaType.SVCD or MediaType.PCD or MediaType.DTSCD or MediaType.CDMIDI or MediaType.CDV or MediaType.CDIREADY or MediaType.FMTOWNS or MediaType.PS1CD or MediaType.PS2CD or MediaType.MEGACD or MediaType.SATURNCD or MediaType.GDROM or MediaType.GDR or MediaType.MilCD or MediaType.SuperCDROM2 or MediaType.JaguarCD or MediaType.ThreeDO or MediaType.PCFX or MediaType.NeoGeoCD or MediaType.CDTV or MediaType.CD32 or MediaType.Playdia or MediaType.Pippin or MediaType.VideoNow or MediaType.VideoNowColor or MediaType.VideoNowXp or MediaType.CVD) return ErrorNumber.NoError; { foreach(Track track in Tracks) { track.Pregap = 0; track.Indexes?.Clear(); } } return ErrorNumber.NoError; } /// public ErrorNumber ReadMediaTag(MediaTagType tag, out byte[] buffer) { buffer = null; return _mediaTags.TryGetValue(tag, out buffer) ? ErrorNumber.NoError : ErrorNumber.NoData; } /// public ErrorNumber ReadSector(ulong sectorAddress, out byte[] buffer) { buffer = null; if(sectorAddress > _imageInfo.Sectors - 1) return ErrorNumber.OutOfRange; ulong ddtEntry = GetDdtEntry(sectorAddress); uint offsetMask = (uint)((1 << _shift) - 1); ulong offset = ddtEntry & offsetMask; ulong blockOffset = ddtEntry >> _shift; // Partially written image... as we can't know the real sector size just assume it's common :/ if(ddtEntry == 0) { buffer = new byte[_imageInfo.SectorSize]; return ErrorNumber.NoError; } // Check if block is cached if(_blockCache.TryGetValue(blockOffset, out byte[] block) && _blockHeaderCache.TryGetValue(blockOffset, out BlockHeader blockHeader)) { buffer = new byte[blockHeader.sectorSize]; Array.Copy(block, (long)(offset * blockHeader.sectorSize), buffer, 0, blockHeader.sectorSize); return ErrorNumber.NoError; } // Read block header _imageStream.Position = (long)blockOffset; _structureBytes = new byte[Marshal.SizeOf()]; _imageStream.EnsureRead(_structureBytes, 0, _structureBytes.Length); blockHeader = Marshal.SpanToStructureLittleEndian(_structureBytes); // Decompress block AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); ulong decompressedLength; switch(blockHeader.compression) { case CompressionType.None: block = new byte[blockHeader.length]; _imageStream.EnsureRead(block, 0, (int)blockHeader.length); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case CompressionType.Lzma: byte[] compressedBlock = new byte[blockHeader.cmpLength - LZMA_PROPERTIES_LENGTH]; byte[] lzmaProperties = new byte[LZMA_PROPERTIES_LENGTH]; _imageStream.EnsureRead(lzmaProperties, 0, LZMA_PROPERTIES_LENGTH); _imageStream.EnsureRead(compressedBlock, 0, compressedBlock.Length); block = new byte[blockHeader.length]; decompressedLength = (ulong)LZMA.DecodeBuffer(compressedBlock, block, lzmaProperties); if(decompressedLength != blockHeader.length) { AaruLogging.Debug(MODULE_NAME, Localization.Error_decompressing_block_should_be_0_bytes_but_got_1_bytes, blockHeader.length, decompressedLength); return ErrorNumber.InOutError; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; case CompressionType.Flac: byte[] flacBlock = new byte[blockHeader.cmpLength]; _imageStream.EnsureRead(flacBlock, 0, flacBlock.Length); block = new byte[blockHeader.length]; decompressedLength = (ulong)FLAC.DecodeBuffer(flacBlock, block); if(decompressedLength != blockHeader.length) { AaruLogging.Debug(MODULE_NAME, Localization.Error_decompressing_block_should_be_0_bytes_but_got_1_bytes, blockHeader.length, decompressedLength); return ErrorNumber.InOutError; } AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); break; default: return ErrorNumber.NotSupported; } // Check if cache needs to be emptied if(_currentCacheSize + blockHeader.length >= MAX_CACHE_SIZE) { _currentCacheSize = 0; _blockHeaderCache = new Dictionary(); _blockCache = new Dictionary(); } // Add block to cache _currentCacheSize += blockHeader.length; _blockHeaderCache.Add(blockOffset, blockHeader); _blockCache.Add(blockOffset, block); buffer = new byte[blockHeader.sectorSize]; Array.Copy(block, (long)(offset * blockHeader.sectorSize), buffer, 0, blockHeader.sectorSize); AaruLogging.Debug(MODULE_NAME, Localization.Memory_snapshot_0_bytes, GC.GetTotalMemory(false)); return ErrorNumber.NoError; } /// public ErrorNumber ReadSectorTag(ulong sectorAddress, SectorTagType tag, out byte[] buffer) => ReadSectorsTag(sectorAddress, 1, tag, out buffer); /// public ErrorNumber ReadSector(ulong sectorAddress, uint track, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); return trk?.Sequence != track ? ErrorNumber.SectorNotFound : ReadSector(trk.StartSector + sectorAddress, out buffer); } /// public ErrorNumber ReadSectorTag(ulong sectorAddress, uint track, SectorTagType tag, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); return trk?.Sequence != track ? ErrorNumber.SectorNotFound : ReadSectorTag(trk.StartSector + sectorAddress, tag, out buffer); } /// public ErrorNumber ReadSectors(ulong sectorAddress, uint length, out byte[] buffer) { buffer = null; if(sectorAddress > _imageInfo.Sectors - 1) return ErrorNumber.OutOfRange; if(sectorAddress + length > _imageInfo.Sectors) return ErrorNumber.OutOfRange; var ms = new MemoryStream(); for(uint i = 0; i < length; i++) { ErrorNumber errno = ReadSector(sectorAddress + i, out byte[] sector); if(errno != ErrorNumber.NoError) return errno; ms.Write(sector, 0, sector.Length); } buffer = ms.ToArray(); return ErrorNumber.NoError; } /// public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, SectorTagType tag, out byte[] buffer) { uint sectorOffset; uint sectorSize; uint sectorSkip; byte[] dataSource; buffer = null; if(_imageInfo.MetadataMediaType == MetadataMediaType.OpticalDisc) { Track trk = Tracks.FirstOrDefault(t => sectorAddress >= t.StartSector && sectorAddress <= t.EndSector); if(trk is null) return ErrorNumber.SectorNotFound; if(trk.Sequence == 0 && trk.StartSector == 0 && trk.EndSector == 0) return ErrorNumber.SectorNotFound; switch(tag) { case SectorTagType.CdSectorEcc: case SectorTagType.CdSectorEccP: case SectorTagType.CdSectorEccQ: case SectorTagType.CdSectorEdc: case SectorTagType.CdSectorHeader: case SectorTagType.CdSectorSubchannel: case SectorTagType.CdSectorSubHeader: case SectorTagType.CdSectorSync: case SectorTagType.DvdSectorCmi: case SectorTagType.DvdSectorTitleKey: case SectorTagType.DvdSectorInformation: case SectorTagType.DvdSectorNumber: case SectorTagType.DvdSectorIed: case SectorTagType.DvdSectorEdc: case SectorTagType.DvdTitleKeyDecrypted: break; case SectorTagType.CdTrackFlags: if(!_trackFlags.TryGetValue((byte)sectorAddress, out byte flags)) return ErrorNumber.NoData; buffer = [flags]; return ErrorNumber.NoError; case SectorTagType.CdTrackIsrc: if(!_trackIsrcs.TryGetValue((byte)sectorAddress, out string isrc)) return ErrorNumber.NoData; buffer = Encoding.UTF8.GetBytes(isrc); return ErrorNumber.NoError; default: return ErrorNumber.NotSupported; } switch(trk.Type) { case TrackType.CdMode1: switch(tag) { case SectorTagType.CdSectorSync: { sectorOffset = 0; sectorSize = 12; sectorSkip = 4; dataSource = _sectorPrefix; break; } case SectorTagType.CdSectorHeader: { sectorOffset = 12; sectorSize = 4; sectorSkip = 2336; dataSource = _sectorPrefix; break; } case SectorTagType.CdSectorSubHeader: return ErrorNumber.NotSupported; case SectorTagType.CdSectorEcc: { sectorOffset = 12; sectorSize = 276; sectorSkip = 0; dataSource = _sectorSuffix; break; } case SectorTagType.CdSectorEccP: { sectorOffset = 12; sectorSize = 172; sectorSkip = 104; dataSource = _sectorSuffix; break; } case SectorTagType.CdSectorEccQ: { sectorOffset = 184; sectorSize = 104; sectorSkip = 0; dataSource = _sectorSuffix; break; } case SectorTagType.CdSectorEdc: { sectorOffset = 0; sectorSize = 4; sectorSkip = 284; dataSource = _sectorSuffix; break; } case SectorTagType.CdSectorSubchannel: { sectorOffset = 0; sectorSize = 96; sectorSkip = 0; dataSource = _sectorSubchannel; break; } default: return ErrorNumber.NotSupported; } break; case TrackType.CdMode2Formless: case TrackType.CdMode2Form1: case TrackType.CdMode2Form2: { switch(tag) { case SectorTagType.CdSectorSync: { sectorOffset = 0; sectorSize = 12; sectorSkip = 4; dataSource = _sectorPrefix; break; } case SectorTagType.CdSectorHeader: { sectorOffset = 12; sectorSize = 4; sectorSkip = 0; dataSource = _sectorPrefix; break; } case SectorTagType.CdSectorSubHeader: { sectorOffset = 0; sectorSize = 8; sectorSkip = 0; dataSource = _mode2Subheaders; break; } // These could be implemented case SectorTagType.CdSectorEcc: case SectorTagType.CdSectorEccP: case SectorTagType.CdSectorEccQ: case SectorTagType.CdSectorEdc: return ErrorNumber.NotSupported; case SectorTagType.CdSectorSubchannel: { sectorOffset = 0; sectorSize = 96; sectorSkip = 0; dataSource = _sectorSubchannel; break; } default: return ErrorNumber.NotSupported; } break; } case TrackType.Audio: { switch(tag) { case SectorTagType.CdSectorSubchannel: { sectorOffset = 0; sectorSize = 96; sectorSkip = 0; dataSource = _sectorSubchannel; break; } default: return ErrorNumber.NotSupported; } break; } case TrackType.Data: { if(_imageInfo.MediaType == MediaType.DVDROM) { switch(tag) { case SectorTagType.DvdSectorCmi: { sectorOffset = 0; sectorSize = 1; sectorSkip = 5; dataSource = _sectorCprMai; break; } case SectorTagType.DvdSectorTitleKey: { sectorOffset = 1; sectorSize = 5; sectorSkip = 0; dataSource = _sectorCprMai; break; } case SectorTagType.DvdSectorInformation: { sectorOffset = 0; sectorSize = 1; sectorSkip = 3; dataSource = _sectorId; break; } case SectorTagType.DvdSectorNumber: { sectorOffset = 1; sectorSize = 3; sectorSkip = 0; dataSource = _sectorId; break; } case SectorTagType.DvdSectorIed: { sectorOffset = 0; sectorSize = 2; sectorSkip = 0; dataSource = _sectorIed; break; } case SectorTagType.DvdSectorEdc: { sectorOffset = 0; sectorSize = 4; sectorSkip = 0; dataSource = _sectorEdc; break; } case SectorTagType.DvdTitleKeyDecrypted: { sectorOffset = 0; sectorSize = 5; sectorSkip = 0; dataSource = _sectorDecryptedTitleKey; break; } default: return ErrorNumber.NotSupported; } } else return ErrorNumber.NotSupported; break; } default: return ErrorNumber.NotSupported; } } else return ErrorNumber.NoData; if(dataSource == null) return ErrorNumber.NotSupported; buffer = new byte[sectorSize * length]; if(sectorOffset == 0 && sectorSkip == 0) { Array.Copy(dataSource, (long)(sectorAddress * sectorSize), buffer, 0, length * sectorSize); return ErrorNumber.NoError; } for(int i = 0; i < length; i++) { Array.Copy(dataSource, (long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip) + sectorOffset), buffer, i * sectorSize, sectorSize); } return ErrorNumber.NoError; } /// public ErrorNumber ReadSectors(ulong sectorAddress, uint length, uint track, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); if(trk?.Sequence != track) return ErrorNumber.SectorNotFound; return trk.StartSector + sectorAddress + length > trk.EndSector + 1 ? ErrorNumber.OutOfRange : ReadSectors(trk.StartSector + sectorAddress, length, out buffer); } /// public ErrorNumber ReadSectorsTag(ulong sectorAddress, uint length, uint track, SectorTagType tag, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); return trk?.Sequence != track ? ErrorNumber.SectorNotFound : trk.StartSector + sectorAddress + length > trk.EndSector + 1 ? ErrorNumber.OutOfRange : ReadSectorsTag(trk.StartSector + sectorAddress, length, tag, out buffer); } /// public ErrorNumber ReadSectorLong(ulong sectorAddress, out byte[] buffer) { buffer = null; switch(_imageInfo.MetadataMediaType) { case MetadataMediaType.OpticalDisc: Track trk = Tracks.FirstOrDefault(t => sectorAddress >= t.StartSector && sectorAddress <= t.EndSector); if(trk is null) return ErrorNumber.SectorNotFound; if(trk.Sequence == 0 && trk.StartSector == 0 && trk.EndSector == 0) return ErrorNumber.SectorNotFound; if((_sectorSuffix == null || _sectorPrefix == null) && (_sectorSuffixDdt == null || _sectorPrefixDdt == null)) return ReadSector(sectorAddress, out buffer); buffer = new byte[2352]; ErrorNumber errno = ReadSector(sectorAddress, out byte[] data); if(errno != ErrorNumber.NoError) return errno; switch(trk.Type) { case TrackType.Audio: case TrackType.Data: buffer = data; return ErrorNumber.NoError; case TrackType.CdMode1: Array.Copy(data, 0, buffer, 16, 2048); if(_sectorPrefix != null) Array.Copy(_sectorPrefix, (int)sectorAddress * 16, buffer, 0, 16); else if(_sectorPrefixDdt != null) { if((_sectorPrefixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.Correct) ReconstructPrefix(ref buffer, trk.Type, (long)sectorAddress); else if((_sectorPrefixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.NotDumped || _sectorPrefixDdt[sectorAddress] == 0) { // Do nothing } else { uint prefixPosition = ((_sectorPrefixDdt[sectorAddress] & CD_DFIX_MASK) - 1) * 16; if(prefixPosition > _sectorPrefixMs.Length) return ErrorNumber.InvalidArgument; _sectorPrefixMs.Position = prefixPosition; _sectorPrefixMs.EnsureRead(buffer, 0, 16); } } else return ErrorNumber.InvalidArgument; if(_sectorSuffix != null) Array.Copy(_sectorSuffix, (int)sectorAddress * 288, buffer, 2064, 288); else if(_sectorSuffixDdt != null) { if((_sectorSuffixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.Correct) ReconstructEcc(ref buffer, trk.Type); else if((_sectorSuffixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.NotDumped || _sectorSuffixDdt[sectorAddress] == 0) { // Do nothing } else { uint suffixPosition = ((_sectorSuffixDdt[sectorAddress] & CD_DFIX_MASK) - 1) * 288; if(suffixPosition > _sectorSuffixMs.Length) return ErrorNumber.InvalidArgument; _sectorSuffixMs.Position = suffixPosition; _sectorSuffixMs.EnsureRead(buffer, 2064, 288); } } else return ErrorNumber.InvalidArgument; return ErrorNumber.NoError; case TrackType.CdMode2Formless: case TrackType.CdMode2Form1: case TrackType.CdMode2Form2: if(_sectorPrefix != null) Array.Copy(_sectorPrefix, (int)sectorAddress * 16, buffer, 0, 16); else if(_sectorPrefixMs != null) { if((_sectorPrefixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.Correct) ReconstructPrefix(ref buffer, trk.Type, (long)sectorAddress); else if((_sectorPrefixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.NotDumped || _sectorPrefixDdt[sectorAddress] == 0) { // Do nothing } else { uint prefixPosition = ((_sectorPrefixDdt[sectorAddress] & CD_DFIX_MASK) - 1) * 16; if(prefixPosition > _sectorPrefixMs.Length) return ErrorNumber.InvalidArgument; _sectorPrefixMs.Position = prefixPosition; _sectorPrefixMs.EnsureRead(buffer, 0, 16); } } else return ErrorNumber.InvalidArgument; if(_mode2Subheaders != null && _sectorSuffixDdt != null) { Array.Copy(_mode2Subheaders, (int)sectorAddress * 8, buffer, 16, 8); switch(_sectorSuffixDdt[sectorAddress] & CD_XFIX_MASK) { case (uint)CdFixFlags.Mode2Form1Ok: Array.Copy(data, 0, buffer, 24, 2048); ReconstructEcc(ref buffer, TrackType.CdMode2Form1); break; case (uint)CdFixFlags.Mode2Form2Ok: case (uint)CdFixFlags.Mode2Form2NoCrc: { Array.Copy(data, 0, buffer, 24, 2324); if((_sectorSuffixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.Mode2Form2Ok) ReconstructEcc(ref buffer, TrackType.CdMode2Form2); break; } default: { if((_sectorSuffixDdt[sectorAddress] & CD_XFIX_MASK) == (uint)CdFixFlags.NotDumped || _sectorSuffixDdt[sectorAddress] == 0) { // Do nothing } else // Mode 2 where EDC failed { // Incorrectly written images if(data.Length == 2328) Array.Copy(data, 0, buffer, 24, 2328); else { bool form2 = (buffer[18] & 0x20) == 0x20 || (buffer[22] & 0x20) == 0x20; uint suffixPosition = ((_sectorSuffixDdt[sectorAddress] & CD_DFIX_MASK) - 1) * 288; if(suffixPosition > _sectorSuffixMs.Length) return ErrorNumber.InvalidArgument; _sectorSuffixMs.Position = suffixPosition; _sectorSuffixMs.EnsureRead(buffer, form2 ? 2348 : 2072, form2 ? 4 : 280); Array.Copy(data, 0, buffer, 24, form2 ? 2324 : 2048); } } break; } } } else if(_mode2Subheaders != null) { Array.Copy(_mode2Subheaders, (int)sectorAddress * 8, buffer, 16, 8); Array.Copy(data, 0, buffer, 24, 2328); } else Array.Copy(data, 0, buffer, 16, 2336); return ErrorNumber.NoError; } break; case MetadataMediaType.BlockMedia: switch(_imageInfo.MediaType) { case MediaType.AppleFileWare: case MediaType.AppleProfile: case MediaType.AppleSonySS: case MediaType.AppleSonyDS: case MediaType.AppleWidget: case MediaType.PriamDataTower: return ReadSectorsLong(sectorAddress, 1, out buffer); } break; } return ErrorNumber.NotSupported; } /// public ErrorNumber ReadSectorLong(ulong sectorAddress, uint track, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); return trk?.Sequence != track ? ErrorNumber.SectorNotFound : ReadSectorLong(trk.StartSector + sectorAddress, out buffer); } /// public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, out byte[] buffer) { byte[] data; ErrorNumber errno; buffer = null; switch(_imageInfo.MetadataMediaType) { case MetadataMediaType.OpticalDisc: Track trk = Tracks.FirstOrDefault(t => sectorAddress >= t.StartSector && sectorAddress <= t.EndSector); if(trk is null) return ErrorNumber.SectorNotFound; if(trk.Sequence == 0 && trk.StartSector == 0 && trk.EndSector == 0) return ErrorNumber.SectorNotFound; if(sectorAddress + length > trk.EndSector + 1) return ErrorNumber.OutOfRange; switch(trk.Type) { // These types only contain user data case TrackType.Audio: case TrackType.Data: return ReadSectors(sectorAddress, length, out buffer); // Join prefix (sync, header) with user data with suffix (edc, ecc p, ecc q) case TrackType.CdMode1: if(_sectorPrefix != null && _sectorSuffix != null) { buffer = new byte[2352 * length]; errno = ReadSectors(sectorAddress, length, out data); if(errno != ErrorNumber.NoError) return errno; for(uint i = 0; i < length; i++) { Array.Copy(_sectorPrefix, (int)((sectorAddress + i) * 16), buffer, (int)(i * 2352), 16); Array.Copy(data, (int)(i * 2048), buffer, (int)(i * 2352) + 16, 2048); Array.Copy(_sectorSuffix, (int)((sectorAddress + i) * 288), buffer, (int)(i * 2352) + 2064, 288); } return ErrorNumber.NoError; } if(_sectorPrefixDdt == null || _sectorSuffixDdt == null) return ReadSectors(sectorAddress, length, out buffer); buffer = new byte[2352 * length]; for(uint i = 0; i < length; i++) { errno = ReadSectorLong(sectorAddress + i, out byte[] temp); if(errno != ErrorNumber.NoError) return errno; Array.Copy(temp, 0, buffer, 2352 * i, 2352); } return ErrorNumber.NoError; // Join prefix (sync, header) with user data case TrackType.CdMode2Formless: case TrackType.CdMode2Form1: case TrackType.CdMode2Form2: if(_sectorPrefix != null && _sectorSuffix != null) { buffer = new byte[2352 * length]; errno = ReadSectors(sectorAddress, length, out data); if(errno != ErrorNumber.NoError) return errno; for(uint i = 0; i < length; i++) { Array.Copy(_sectorPrefix, (int)((sectorAddress + i) * 16), buffer, (int)(i * 2352), 16); Array.Copy(data, (int)(i * 2336), buffer, (int)(i * 2352) + 16, 2336); } return ErrorNumber.NoError; } if(_sectorPrefixDdt == null || _sectorSuffixDdt == null) return ReadSectors(sectorAddress, length, out buffer); buffer = new byte[2352 * length]; for(uint i = 0; i < length; i++) { errno = ReadSectorLong(sectorAddress + i, out byte[] temp); if(errno != ErrorNumber.NoError) return errno; Array.Copy(temp, 0, buffer, 2352 * i, 2352); } return ErrorNumber.NoError; } break; case MetadataMediaType.BlockMedia: switch(_imageInfo.MediaType) { // Join user data with tags case MediaType.AppleFileWare: case MediaType.AppleProfile: case MediaType.AppleSonySS: case MediaType.AppleSonyDS: case MediaType.AppleWidget: case MediaType.PriamDataTower: if(_sectorSubchannel == null) return ReadSector(sectorAddress, out buffer); uint tagSize = _imageInfo.MediaType switch { MediaType.AppleFileWare or MediaType.AppleProfile or MediaType.AppleWidget => 20, MediaType.AppleSonySS or MediaType.AppleSonyDS => 12, MediaType.PriamDataTower => 24, _ => 0 }; uint sectorSize = 512 + tagSize; errno = ReadSectors(sectorAddress, length, out data); if(errno != ErrorNumber.NoError) return errno; buffer = new byte[(sectorSize + 512) * length]; for(uint i = 0; i < length; i++) { Array.Copy(_sectorSubchannel, (int)((sectorAddress + i) * tagSize), buffer, (int)(i * sectorSize + 512), tagSize); Array.Copy(data, (int)((sectorAddress + i) * 512), buffer, (int)(i * 512), 512); } return ErrorNumber.NoError; } break; } return ErrorNumber.NotSupported; } /// public ErrorNumber ReadSectorsLong(ulong sectorAddress, uint length, uint track, out byte[] buffer) { buffer = null; if(_imageInfo.MetadataMediaType != MetadataMediaType.OpticalDisc) return ErrorNumber.NotSupported; Track trk = Tracks.FirstOrDefault(t => t.Sequence == track); return trk?.Sequence != track ? ErrorNumber.SectorNotFound : trk.StartSector + sectorAddress + length > trk.EndSector + 1 ? ErrorNumber.OutOfRange : ReadSectorsLong(trk.StartSector + sectorAddress, length, out buffer); } /// public List GetSessionTracks(Session session) => Tracks.Where(t => t.Sequence == session.Sequence).ToList(); /// public List GetSessionTracks(ushort session) => Tracks.Where(t => t.Sequence == session).ToList(); #endregion }