Files
Aaru/Aaru.Images/BlindWrite4/Read.cs

1243 lines
60 KiB
C#
Raw Normal View History

// /***************************************************************************
2020-02-27 12:31:25 +00:00
// Aaru Data Preservation Suite
// ----------------------------------------------------------------------------
//
// Filename : Read.cs
// Author(s) : Natalia Portillo <claunia@claunia.com>
//
// Component : Disk image plugins.
//
// --[ Description ] ----------------------------------------------------------
//
// Reads BlindWrite 4 disc images.
//
// --[ License ] --------------------------------------------------------------
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation; either version 2.1 of the
// License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, see <http://www.gnu.org/licenses/>.
//
// ----------------------------------------------------------------------------
2020-12-31 23:08:23 +00:00
// Copyright © 2011-2021 Natalia Portillo
// ****************************************************************************/
using System;
using System.Collections.Generic;
2017-12-19 19:33:46 +00:00
using System.Globalization;
using System.IO;
using System.Linq;
using System.Text;
2020-02-27 00:33:26 +00:00
using Aaru.CommonTypes;
using Aaru.CommonTypes.Enums;
using Aaru.CommonTypes.Exceptions;
using Aaru.CommonTypes.Interfaces;
using Aaru.CommonTypes.Structs;
using Aaru.Console;
using Aaru.Decoders.CD;
2020-07-20 15:43:52 +01:00
using Aaru.Helpers;
2020-02-27 00:33:26 +00:00
using Session = Aaru.CommonTypes.Structs.Session;
2020-02-27 00:33:26 +00:00
namespace Aaru.DiscImages
{
2020-07-22 13:20:25 +01:00
public sealed partial class BlindWrite4
{
public bool Open(IFilter imageFilter)
{
* TODO: * README.md: * DiscImageChef.sln: * DiscImageChef/Commands/Ls.cs: * DiscImageChef.Filters/GZip.cs: * DiscImageChef.DiscImages/BLU.cs: * DiscImageChef.DiscImages/VHD.cs: * DiscImageChef.DiscImages/VDI.cs: * DiscImageChef.DiscImages/QED.cs: * DiscImageChef.DiscImages/DIM.cs: * DiscImageChef.DiscImages/GDI.cs: * DiscImageChef.Filters/Filter.cs: * DiscImageChef/Commands/Decode.cs: * DiscImageChef.DiscImages/QCOW.cs: * DiscImageChef.Filters/Filters.cs: * DiscImageChef/Core/Statistics.cs: * DiscImageChef.DiscImages/VHDX.cs: * DiscImageChef.DiscImages/Nero.cs: * DiscImageChef/Commands/Verify.cs: * DiscImageChef.DiscImages/UDIF.cs: * DiscImageChef/Commands/Compare.cs: * DiscImageChef/Commands/Analyze.cs: * DiscImageChef.DiscImages/QCOW2.cs: * DiscImageChef/Commands/Entropy.cs: * DiscImageChef/Commands/Formats.cs: * DiscImageChef/Commands/PrintHex.cs: * DiscImageChef.DiscImages/VMware.cs: * DiscImageChef.Settings/Settings.cs: * DiscImageChef/DetectImageFormat.cs: * DiscImageChef/DiscImageChef.csproj: * DiscImageChef.DiscImages/CDRDAO.cs: * DiscImageChef.DiscImages/CPCDSK.cs: * DiscImageChef/Commands/Checksum.cs: * DiscImageChef.DiscImages/CopyQM.cs: * DiscImageChef.DiscImages/CDRWin.cs: * DiscImageChef/Commands/Configure.cs: * DiscImageChef/Commands/DumpMedia.cs: * DiscImageChef/Commands/Statistics.cs: * DiscImageChef.Filters/ZZZNoFilter.cs: * DiscImageChef.DiscImages/TeleDisk.cs: * DiscImageChef.DiscImages/Apple2MG.cs: * DiscImageChef.Filters/OffsetStream.cs: * DiscImageChef.DiscImages/Parallels.cs: * DiscImageChef/Commands/ExtractFiles.cs: * DiscImageChef.DiscImages/DiskCopy42.cs: * DiscImageChef.DiscImages/Alcohol120.cs: * DiscImageChef.DiscImages/ZZZRawImage.cs: * DiscImageChef/Commands/CreateSidecar.cs: * DiscImageChef.DiscImages/ImagePlugin.cs: * DiscImageChef.DiscImages/BlindWrite5.cs: * DiscImageChef.DiscImages/BlindWrite4.cs: * DiscImageChef.Filters/ForcedSeekStream.cs: * DiscImageChef.Filters/Properties/AssemblyInfo.cs: * DiscImageChef.Filters/DiscImageChef.Filters.csproj: * DiscImageChef.DiscImages/DiscImageChef.DiscImages.csproj: Added filters. * DiscImageChef.Filesystems/CPM/Info.cs: Do not throw identification exceptions. * DiscImageChef/Plugins.cs: Sorted plugins lists.
2016-09-05 17:37:31 +01:00
Stream stream = imageFilter.GetDataForkStream();
stream.Seek(0, SeekOrigin.Begin);
if(stream.Length < 19)
return false;
byte[] tmpArray = new byte[19];
byte[] tmpUShort = new byte[2];
byte[] tmpUInt = new byte[4];
byte[] tmpULong = new byte[8];
stream.Read(tmpArray, 0, 19);
2020-07-20 21:11:32 +01:00
if(!_bw4Signature.SequenceEqual(tmpArray))
return false;
_header = new Header
{
Signature = tmpArray
};
// Seems to always be 2
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.Unknown1 = BitConverter.ToUInt32(tmpUInt, 0);
// Seems to be a timetamp
stream.Read(tmpULong, 0, 8);
2020-07-20 21:11:32 +01:00
_header.Timestamp = BitConverter.ToUInt64(tmpULong, 0);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.VolumeIdLength = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[_header.VolumeIdLength];
stream.Read(tmpArray, 0, tmpArray.Length);
2020-07-20 21:11:32 +01:00
_header.VolumeIdBytes = tmpArray;
_header.VolumeIdentifier = StringHandlers.CToString(_header.VolumeIdBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.SysIdLength = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[_header.SysIdLength];
stream.Read(tmpArray, 0, tmpArray.Length);
2020-07-20 21:11:32 +01:00
_header.SysIdBytes = tmpArray;
_header.SystemIdentifier = StringHandlers.CToString(_header.SysIdBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.CommentsLength = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[_header.CommentsLength];
stream.Read(tmpArray, 0, tmpArray.Length);
2020-07-20 21:11:32 +01:00
_header.CommentsBytes = tmpArray;
_header.Comments = StringHandlers.CToString(_header.CommentsBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.TrackDescriptors = BitConverter.ToUInt32(tmpUInt, 0);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.DataFileLength = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[_header.DataFileLength];
stream.Read(tmpArray, 0, tmpArray.Length);
2020-07-20 21:11:32 +01:00
_header.DataFileBytes = tmpArray;
_header.DataFile = StringHandlers.CToString(_header.DataFileBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.SubchannelFileLength = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[_header.SubchannelFileLength];
stream.Read(tmpArray, 0, tmpArray.Length);
2020-07-20 21:11:32 +01:00
_header.SubchannelFileBytes = tmpArray;
_header.SubchannelFile = StringHandlers.CToString(_header.SubchannelFileBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
2020-07-20 21:11:32 +01:00
_header.Unknown2 = BitConverter.ToUInt32(tmpUInt, 0);
_header.Unknown3 = (byte)stream.ReadByte();
tmpArray = new byte[_header.Unknown3];
stream.Read(tmpArray, 0, _header.Unknown3);
_header.Unknown4 = tmpArray;
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.signature = {0}",
2020-07-20 21:11:32 +01:00
StringHandlers.CToString(_header.Signature));
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.unknown1 = {0}", _header.Unknown1);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.timestamp = {0}", _header.Timestamp);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.volumeIdLength = {0}", _header.VolumeIdLength);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.volumeIdentifier = {0}", _header.VolumeIdentifier);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.sysIdLength = {0}", _header.SysIdLength);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.systemIdentifier = {0}", _header.SystemIdentifier);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.commentsLength = {0}", _header.CommentsLength);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.comments = {0}", _header.Comments);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.trackDescriptors = {0}", _header.TrackDescriptors);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.dataFileLength = {0}", _header.DataFileLength);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.dataFilter = {0}", _header.DataFilter);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.dataFile = {0}", _header.DataFile);
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.subchannelFileLength = {0}",
2020-07-20 21:11:32 +01:00
_header.SubchannelFileLength);
2020-07-20 21:11:32 +01:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.subchannelFilter = {0}", _header.SubchannelFilter);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.subchannelFile = {0}", _header.SubchannelFile);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.unknown2 = {0}", _header.Unknown2);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.unknown3 = {0}", _header.Unknown3);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "header.unknown4.Length = {0}", _header.Unknown4.Length);
_bwTracks = new List<TrackDescriptor>();
2020-07-20 21:11:32 +01:00
for(int i = 0; i < _header.TrackDescriptors; i++)
{
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "stream.Position = {0}", stream.Position);
var track = new TrackDescriptor();
stream.Read(tmpUInt, 0, 4);
track.filenameLen = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.filenameLen];
stream.Read(tmpArray, 0, tmpArray.Length);
track.filenameBytes = tmpArray;
track.filename = StringHandlers.CToString(track.filenameBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.offset = BitConverter.ToUInt32(tmpUInt, 0);
track.subchannel = (byte)stream.ReadByte();
tmpArray = new byte[3];
stream.Read(tmpArray, 0, 3);
track.unknown1 = tmpArray;
stream.Read(tmpUInt, 0, 4);
track.unknown2 = BitConverter.ToUInt32(tmpUInt, 0);
track.unknown3 = (byte)stream.ReadByte();
track.session = (byte)stream.ReadByte();
track.unknown4 = (byte)stream.ReadByte();
track.adrCtl = (byte)stream.ReadByte();
track.unknown5 = (byte)stream.ReadByte();
track.trackMode = (TrackType)stream.ReadByte();
track.unknown6 = (byte)stream.ReadByte();
track.point = (byte)stream.ReadByte();
stream.Read(tmpUInt, 0, 4);
track.unknown7 = BitConverter.ToUInt32(tmpUInt, 0);
stream.Read(tmpUInt, 0, 4);
track.unknown8 = BitConverter.ToUInt32(tmpUInt, 0);
stream.Read(tmpUInt, 0, 4);
track.pregapOffsetAdjustment = BitConverter.ToUInt32(tmpUInt, 0);
stream.Read(tmpUInt, 0, 4);
track.unknown10 = BitConverter.ToUInt32(tmpUInt, 0);
stream.Read(tmpUShort, 0, 2);
track.unknown11 = BitConverter.ToUInt16(tmpUShort, 0);
stream.Read(tmpUInt, 0, 4);
track.lastSector = BitConverter.ToUInt32(tmpUInt, 0);
track.unknown12 = (byte)stream.ReadByte();
// This is off by one
track.lastSector--;
stream.Read(tmpUInt, 0, 4);
track.pregap = BitConverter.ToInt32(tmpUInt, 0);
stream.Read(tmpUInt, 0, 4);
track.startSector = BitConverter.ToInt32(tmpUInt, 0);
track.unknown13 = new uint[2];
for(int j = 0; j < track.unknown13.Length; j++)
{
stream.Read(tmpUInt, 0, 4);
track.unknown13[j] = BitConverter.ToUInt32(tmpUInt, 0);
}
stream.Read(tmpUInt, 0, 4);
track.titleLen = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.titleLen];
stream.Read(tmpArray, 0, tmpArray.Length);
track.titleBytes = tmpArray;
track.title = StringHandlers.CToString(track.titleBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.performerLen = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.performerLen];
stream.Read(tmpArray, 0, tmpArray.Length);
track.performerBytes = tmpArray;
track.performer = StringHandlers.CToString(track.performerBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen1 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen1];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes1 = tmpArray;
track.unkString1 = StringHandlers.CToString(track.unkStrBytes1, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen2 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen2];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes2 = tmpArray;
track.unkString2 = StringHandlers.CToString(track.unkStrBytes2, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen3 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen3];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes3 = tmpArray;
track.unkString3 = StringHandlers.CToString(track.unkStrBytes3, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen4 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen4];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes4 = tmpArray;
track.unkString4 = StringHandlers.CToString(track.unkStrBytes4, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.discIdLen = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.discIdLen];
stream.Read(tmpArray, 0, tmpArray.Length);
track.discIdBytes = tmpArray;
track.discId = StringHandlers.CToString(track.discIdBytes, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen5 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen5];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes5 = tmpArray;
track.unkString5 = StringHandlers.CToString(track.unkStrBytes5, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen6 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen6];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes6 = tmpArray;
track.unkString6 = StringHandlers.CToString(track.unkStrBytes6, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen7 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen7];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes7 = tmpArray;
track.unkString7 = StringHandlers.CToString(track.unkStrBytes7, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen8 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen8];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes8 = tmpArray;
track.unkString8 = StringHandlers.CToString(track.unkStrBytes8, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen9 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen9];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes9 = tmpArray;
track.unkString9 = StringHandlers.CToString(track.unkStrBytes9, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen10 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen10];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes10 = tmpArray;
track.unkString10 = StringHandlers.CToString(track.unkStrBytes10, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.unkStrLen11 = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.unkStrLen11];
stream.Read(tmpArray, 0, tmpArray.Length);
track.unkStrBytes11 = tmpArray;
track.unkString11 = StringHandlers.CToString(track.unkStrBytes11, Encoding.Default);
stream.Read(tmpUInt, 0, 4);
track.isrcLen = BitConverter.ToUInt32(tmpUInt, 0);
tmpArray = new byte[track.isrcLen];
stream.Read(tmpArray, 0, tmpArray.Length);
track.isrcBytes = tmpArray;
track.isrcUpc = StringHandlers.CToString(track.isrcBytes, Encoding.Default);
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.filenameLen = {0}", track.filenameLen);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.filename = {0}", track.filename);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.offset = {0}", track.offset);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.subchannel = {0}", track.subchannel);
for(int j = 0; j < track.unknown1.Length; j++)
2020-02-29 18:03:35 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown1[{1}] = 0x{0:X8}",
track.unknown1[j], j);
2017-12-19 20:33:03 +00:00
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown2 = {0}", track.unknown2);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown3 = {0}", track.unknown3);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.session = {0}", track.session);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown4 = {0}", track.unknown4);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.adrCtl = {0}", track.adrCtl);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown5 = {0}", track.unknown5);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.trackMode = {0}", track.trackMode);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown6 = {0}", track.unknown6);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.point = {0}", track.point);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown7 = {0}", track.unknown7);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown8 = {0}", track.unknown8);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown9 = {0}", track.pregapOffsetAdjustment);
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown10 = {0}", track.unknown10);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown11 = {0}", track.unknown11);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.lastSector = {0}", track.lastSector);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown12 = {0}", track.unknown12);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.pregap = {0}", track.pregap);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.startSector = {0}", track.startSector);
for(int j = 0; j < track.unknown13.Length; j++)
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unknown13[{1}] = 0x{0:X8}",
2020-02-29 18:03:35 +00:00
track.unknown13[j], j);
2017-12-19 20:33:03 +00:00
2020-02-27 23:48:41 +00:00
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.titleLen = {0}", track.titleLen);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.title = {0}", track.title);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.performerLen = {0}", track.performerLen);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.performer = {0}", track.performer);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen1 = {0}", track.unkStrLen1);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString1 = {0}", track.unkString1);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen2 = {0}", track.unkStrLen2);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString2 = {0}", track.unkString2);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen3 = {0}", track.unkStrLen3);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString3 = {0}", track.unkString3);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen4 = {0}", track.unkStrLen4);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString4 = {0}", track.unkString4);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.discIdLen = {0}", track.discIdLen);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.discId = {0}", track.discId);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen5 = {0}", track.unkStrLen5);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString5 = {0}", track.unkString5);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen6 = {0}", track.unkStrLen6);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString6 = {0}", track.unkString6);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen7 = {0}", track.unkStrLen7);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString7 = {0}", track.unkString7);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen8 = {0}", track.unkStrLen8);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString8 = {0}", track.unkString8);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen9 = {0}", track.unkStrLen9);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString9 = {0}", track.unkString9);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen10 = {0}", track.unkStrLen10);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString10 = {0}", track.unkString10);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkStrLen11 = {0}", track.unkStrLen11);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.unkString11 = {0}", track.unkString11);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.isrcLen = {0}", track.isrcLen);
AaruConsole.DebugWriteLine("BlindWrite4 plugin", "track.isrcUpc = {0}", track.isrcUpc);
2020-07-20 21:11:32 +01:00
_bwTracks.Add(track);
}
var filtersList = new FiltersList();
* TODO: * README.md: * DiscImageChef.sln: * DiscImageChef/Commands/Ls.cs: * DiscImageChef.Filters/GZip.cs: * DiscImageChef.DiscImages/BLU.cs: * DiscImageChef.DiscImages/VHD.cs: * DiscImageChef.DiscImages/VDI.cs: * DiscImageChef.DiscImages/QED.cs: * DiscImageChef.DiscImages/DIM.cs: * DiscImageChef.DiscImages/GDI.cs: * DiscImageChef.Filters/Filter.cs: * DiscImageChef/Commands/Decode.cs: * DiscImageChef.DiscImages/QCOW.cs: * DiscImageChef.Filters/Filters.cs: * DiscImageChef/Core/Statistics.cs: * DiscImageChef.DiscImages/VHDX.cs: * DiscImageChef.DiscImages/Nero.cs: * DiscImageChef/Commands/Verify.cs: * DiscImageChef.DiscImages/UDIF.cs: * DiscImageChef/Commands/Compare.cs: * DiscImageChef/Commands/Analyze.cs: * DiscImageChef.DiscImages/QCOW2.cs: * DiscImageChef/Commands/Entropy.cs: * DiscImageChef/Commands/Formats.cs: * DiscImageChef/Commands/PrintHex.cs: * DiscImageChef.DiscImages/VMware.cs: * DiscImageChef.Settings/Settings.cs: * DiscImageChef/DetectImageFormat.cs: * DiscImageChef/DiscImageChef.csproj: * DiscImageChef.DiscImages/CDRDAO.cs: * DiscImageChef.DiscImages/CPCDSK.cs: * DiscImageChef/Commands/Checksum.cs: * DiscImageChef.DiscImages/CopyQM.cs: * DiscImageChef.DiscImages/CDRWin.cs: * DiscImageChef/Commands/Configure.cs: * DiscImageChef/Commands/DumpMedia.cs: * DiscImageChef/Commands/Statistics.cs: * DiscImageChef.Filters/ZZZNoFilter.cs: * DiscImageChef.DiscImages/TeleDisk.cs: * DiscImageChef.DiscImages/Apple2MG.cs: * DiscImageChef.Filters/OffsetStream.cs: * DiscImageChef.DiscImages/Parallels.cs: * DiscImageChef/Commands/ExtractFiles.cs: * DiscImageChef.DiscImages/DiskCopy42.cs: * DiscImageChef.DiscImages/Alcohol120.cs: * DiscImageChef.DiscImages/ZZZRawImage.cs: * DiscImageChef/Commands/CreateSidecar.cs: * DiscImageChef.DiscImages/ImagePlugin.cs: * DiscImageChef.DiscImages/BlindWrite5.cs: * DiscImageChef.DiscImages/BlindWrite4.cs: * DiscImageChef.Filters/ForcedSeekStream.cs: * DiscImageChef.Filters/Properties/AssemblyInfo.cs: * DiscImageChef.Filters/DiscImageChef.Filters.csproj: * DiscImageChef.DiscImages/DiscImageChef.DiscImages.csproj: Added filters. * DiscImageChef.Filesystems/CPM/Info.cs: Do not throw identification exceptions. * DiscImageChef/Plugins.cs: Sorted plugins lists.
2016-09-05 17:37:31 +01:00
2020-07-20 21:11:32 +01:00
if(!string.IsNullOrEmpty(_header.DataFile))
while(true)
{
2020-07-20 21:11:32 +01:00
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.DataFile));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
2020-07-20 21:11:32 +01:00
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
_header.DataFile.ToLower(CultureInfo.
CurrentCulture)));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
2020-07-20 21:11:32 +01:00
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
_header.DataFile.ToUpper(CultureInfo.
CurrentCulture)));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.DataFile.
Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).
Last()));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.DataFile.
Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).
Last().ToLower(CultureInfo.CurrentCulture)));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
_dataFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.DataFile.
Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).
Last().ToUpper(CultureInfo.CurrentCulture)));
2020-07-20 21:11:32 +01:00
if(_dataFilter != null)
break;
2020-07-20 21:11:32 +01:00
throw new ArgumentException($"Data file {_header.DataFile} not found");
}
else
throw new ArgumentException("Unable to find data file");
2020-07-20 21:11:32 +01:00
if(!string.IsNullOrEmpty(_header.SubchannelFile))
{
2017-12-19 20:33:03 +00:00
filtersList = new FiltersList();
2020-07-20 21:11:32 +01:00
_subFilter =
((((filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.SubchannelFile)) ??
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
_header.SubchannelFile.
ToLower(CultureInfo.CurrentCulture)))) ??
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
_header.SubchannelFile.
ToUpper(CultureInfo.CurrentCulture)))) ??
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.SubchannelFile.
Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).Last()))) ??
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.SubchannelFile.
Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).Last().
ToLower(CultureInfo.CurrentCulture)))) ??
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), _header.SubchannelFile.Split(new[]
{
'\\'
}, StringSplitOptions.RemoveEmptyEntries).Last().ToUpper(CultureInfo.CurrentCulture)));
}
2020-07-20 21:11:32 +01:00
Tracks = new List<Track>();
Partitions = new List<Partition>();
_offsetMap = new Dictionary<uint, ulong>();
2020-07-20 21:11:32 +01:00
_trackFlags = new Dictionary<uint, byte>();
ushort maxSession = 0;
foreach(TrackDescriptor bwTrack in _bwTracks)
if(bwTrack.point < 0xA0)
{
var track = new Track
{
2020-07-20 04:34:16 +01:00
TrackDescription = bwTrack.title,
TrackEndSector = bwTrack.lastSector
};
if(!string.IsNullOrEmpty(bwTrack.filename))
do
{
2017-12-19 20:33:03 +00:00
track.TrackFilter =
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(), bwTrack.filename));
if(track.TrackFilter != null)
break;
2017-12-19 20:33:03 +00:00
track.TrackFilter =
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
bwTrack.filename.ToLower(CultureInfo.
CurrentCulture)));
2017-12-19 20:33:03 +00:00
if(track.TrackFilter != null)
break;
2017-12-19 20:33:03 +00:00
track.TrackFilter =
filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
bwTrack.filename.ToUpper(CultureInfo.
CurrentCulture)));
if(track.TrackFilter != null)
break;
track.TrackFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
bwTrack.filename.Split(new[]
{
'\\'
},
StringSplitOptions.
RemoveEmptyEntries).
Last()));
if(track.TrackFilter != null)
break;
track.TrackFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
bwTrack.filename.Split(new[]
{
'\\'
},
StringSplitOptions.
RemoveEmptyEntries).
Last().
ToLower(CultureInfo.CurrentCulture)));
if(track.TrackFilter != null)
break;
track.TrackFilter = filtersList.GetFilter(Path.Combine(imageFilter.GetParentFolder(),
bwTrack.filename.Split(new[]
{
'\\'
},
StringSplitOptions.
RemoveEmptyEntries).
Last().
ToUpper(CultureInfo.CurrentCulture)));
2020-07-20 21:11:32 +01:00
track.TrackFilter = _dataFilter;
} while(true);
else
2020-07-20 21:11:32 +01:00
track.TrackFilter = _dataFilter;
2020-07-20 21:11:32 +01:00
track.TrackFile = _dataFilter.GetFilename();
track.TrackFileOffset = bwTrack.offset + ((150 - bwTrack.pregapOffsetAdjustment) * 2352);
track.TrackSubchannelOffset =
((bwTrack.offset / 2352) + (150 - bwTrack.pregapOffsetAdjustment)) * 96;
if(bwTrack.pregap > 0)
{
track.TrackPregap = (ulong)(bwTrack.startSector - bwTrack.pregap);
track.TrackStartSector = (ulong)bwTrack.pregap;
track.TrackFileOffset -= track.TrackPregap * 2352;
track.TrackSubchannelOffset -= track.TrackPregap * 96;
}
else
{
track.TrackPregap = 0;
track.TrackStartSector = (ulong)bwTrack.startSector;
}
2018-06-22 08:08:38 +01:00
track.TrackFileType = "BINARY";
track.TrackRawBytesPerSector = 2352;
track.TrackSequence = bwTrack.point;
track.TrackSession = bwTrack.session;
if(track.TrackSession > maxSession)
maxSession = track.TrackSession;
2020-07-20 21:11:32 +01:00
track.TrackSubchannelFilter = _subFilter;
track.TrackSubchannelFile = _subFilter?.GetFilename();
2020-07-20 21:11:32 +01:00
if(_subFilter != null &&
bwTrack.subchannel > 0)
{
track.TrackSubchannelType = TrackSubchannelType.Packed;
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubchannel))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubchannel);
}
else
track.TrackSubchannelType = TrackSubchannelType.None;
switch(bwTrack.trackMode)
{
case TrackType.Audio:
track.TrackType = CommonTypes.Enums.TrackType.Audio;
2020-07-20 21:11:32 +01:00
_imageInfo.SectorSize = 2352;
track.TrackBytesPerSector = 2352;
break;
case TrackType.Mode1:
track.TrackType = CommonTypes.Enums.TrackType.CdMode1;
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSubHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSubHeader);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEcc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEcc);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccP))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccP);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEccQ))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEccQ);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorEdc))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorEdc);
2020-07-20 21:11:32 +01:00
if(_imageInfo.SectorSize < 2048)
_imageInfo.SectorSize = 2048;
2018-06-22 08:08:38 +01:00
track.TrackBytesPerSector = 2048;
break;
case TrackType.Mode2:
track.TrackType = CommonTypes.Enums.TrackType.CdMode2Formless;
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorSync))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorSync);
2020-07-20 21:11:32 +01:00
if(!_imageInfo.ReadableSectorTags.Contains(SectorTagType.CdSectorHeader))
_imageInfo.ReadableSectorTags.Add(SectorTagType.CdSectorHeader);
2020-07-20 21:11:32 +01:00
if(_imageInfo.SectorSize < 2336)
_imageInfo.SectorSize = 2336;
2018-06-22 08:08:38 +01:00
track.TrackBytesPerSector = 2336;
break;
default:
track.TrackType = CommonTypes.Enums.TrackType.Data;
track.TrackRawBytesPerSector = 2048;
2020-07-20 21:11:32 +01:00
_imageInfo.SectorSize = 2048;
track.TrackBytesPerSector = 2048;
break;
}
if(bwTrack.pregap > 0)
2020-06-17 21:32:19 +01:00
track.Indexes.Add(0, bwTrack.pregap);
2020-06-17 21:32:19 +01:00
track.Indexes.Add(1, bwTrack.startSector);
var partition = new Partition
{
Description = track.TrackDescription,
Size = (track.TrackEndSector - track.TrackStartSector + 1) * 2352,
Length = track.TrackEndSector - track.TrackStartSector + 1,
2020-07-20 04:34:16 +01:00
Sequence = track.TrackSequence,
Start = track.TrackStartSector + track.TrackPregap
};
partition.Offset = partition.Start * 2352;
partition.Type = track.TrackType.ToString();
Partitions.Add(partition);
Tracks.Add(track);
if(!_offsetMap.ContainsKey(track.TrackSequence))
_offsetMap.Add(track.TrackSequence, track.TrackStartSector);
2020-07-20 21:11:32 +01:00
if(!_trackFlags.ContainsKey(track.TrackSequence))
_trackFlags.Add(track.TrackSequence, (byte)(bwTrack.adrCtl & 0x0F));
if(bwTrack.lastSector > _imageInfo.Sectors)
_imageInfo.Sectors = bwTrack.lastSector + 1;
}
else
{
2020-07-20 21:11:32 +01:00
_imageInfo.MediaBarcode = bwTrack.isrcUpc;
_imageInfo.MediaSerialNumber = bwTrack.discId;
_imageInfo.MediaTitle = bwTrack.title;
2017-12-19 20:33:03 +00:00
if(!string.IsNullOrEmpty(bwTrack.isrcUpc) &&
2020-07-20 21:11:32 +01:00
!_imageInfo.ReadableMediaTags.Contains(MediaTagType.CD_MCN))
_imageInfo.ReadableMediaTags.Add(MediaTagType.CD_MCN);
}
Sessions = new List<Session>();
for(ushort i = 1; i <= maxSession; i++)
{
var session = new Session
{
2020-07-20 04:34:16 +01:00
SessionSequence = i,
StartTrack = uint.MaxValue,
StartSector = uint.MaxValue
};
foreach(Track track in Tracks.Where(track => track.TrackSession == i))
2017-12-21 16:33:16 +00:00
{
if(track.TrackSequence < session.StartTrack)
session.StartTrack = track.TrackSequence;
if(track.TrackSequence > session.EndTrack)
session.StartTrack = track.TrackSequence;
if(track.TrackStartSector < session.StartSector)
session.StartSector = track.TrackStartSector;
if(track.TrackEndSector > session.EndSector)
session.EndSector = track.TrackEndSector;
}
Sessions.Add(session);
}
// As long as subchannel is written for any track, it is present for all tracks
if(Tracks.Any(t => t.TrackSubchannelType == TrackSubchannelType.Packed))
foreach(Track track in Tracks)
track.TrackSubchannelType = TrackSubchannelType.Packed;
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CD;
2020-07-20 21:11:32 +01:00
_imageInfo.Application = "BlindWrite";
_imageInfo.ApplicationVersion = "4";
_imageInfo.Version = "4";
2020-07-20 21:11:32 +01:00
_imageInfo.ImageSize = (ulong)_dataFilter.GetDataForkLength();
_imageInfo.CreationTime = _dataFilter.GetCreationTime();
_imageInfo.LastModificationTime = _dataFilter.GetLastWriteTime();
_imageInfo.XmlMediaType = XmlMediaType.OpticalDisc;
bool data = false;
bool mode2 = false;
bool firstAudio = false;
bool firstData = false;
bool audio = false;
foreach(Track bwTrack in Tracks)
{
// First track is audio
firstAudio |= bwTrack.TrackSequence == 1 && bwTrack.TrackType == CommonTypes.Enums.TrackType.Audio;
// First track is data
firstData |= bwTrack.TrackSequence == 1 && bwTrack.TrackType != CommonTypes.Enums.TrackType.Audio;
// Any non first track is data
data |= bwTrack.TrackSequence != 1 && bwTrack.TrackType != CommonTypes.Enums.TrackType.Audio;
// Any non first track is audio
audio |= bwTrack.TrackSequence != 1 && bwTrack.TrackType == CommonTypes.Enums.TrackType.Audio;
switch(bwTrack.TrackType)
{
case CommonTypes.Enums.TrackType.CdMode2Formless:
mode2 = true;
break;
}
}
if(!data &&
!firstData)
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CDDA;
else if(firstAudio &&
data &&
Sessions.Count > 1 &&
mode2)
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CDPLUS;
else if((firstData && audio) || mode2)
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CDROMXA;
else if(!audio)
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CDROM;
else
2020-07-20 21:11:32 +01:00
_imageInfo.MediaType = MediaType.CD;
2020-07-20 21:11:32 +01:00
_imageInfo.Comments = _header.Comments;
2020-07-20 21:11:32 +01:00
AaruConsole.VerboseWriteLine("BlindWrite image describes a disc of type {0}", _imageInfo.MediaType);
2020-07-20 21:11:32 +01:00
if(!string.IsNullOrEmpty(_imageInfo.Comments))
AaruConsole.VerboseWriteLine("BlindWrite comments: {0}", _imageInfo.Comments);
return true;
}
public byte[] ReadDiskTag(MediaTagType tag)
{
switch(tag)
{
case MediaTagType.CD_MCN:
2017-12-19 20:33:03 +00:00
{
2020-07-20 21:11:32 +01:00
if(_imageInfo.MediaSerialNumber != null)
return Encoding.ASCII.GetBytes(_imageInfo.MediaSerialNumber);
2017-12-19 20:33:03 +00:00
throw new FeatureNotPresentImageException("Image does not contain MCN information.");
}
default:
throw new FeatureSupportedButNotImplementedImageException("Feature not supported by image format");
}
}
2018-12-31 13:17:27 +00:00
public byte[] ReadSector(ulong sectorAddress) => ReadSectors(sectorAddress, 1);
2018-12-31 13:17:27 +00:00
public byte[] ReadSectorTag(ulong sectorAddress, SectorTagType tag) => ReadSectorsTag(sectorAddress, 1, tag);
2018-12-31 13:17:27 +00:00
public byte[] ReadSector(ulong sectorAddress, uint track) => ReadSectors(sectorAddress, 1, track);
2018-12-31 13:17:27 +00:00
public byte[] ReadSectorTag(ulong sectorAddress, uint track, SectorTagType tag) =>
ReadSectorsTag(sectorAddress, 1, track, tag);
public byte[] ReadSectors(ulong sectorAddress, uint length)
{
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap where sectorAddress >= kvp.Value
from track in Tracks where track.TrackSequence == kvp.Key
where sectorAddress - kvp.Value <
track.TrackEndSector - track.TrackStartSector + 1 select kvp)
2017-12-21 16:33:16 +00:00
return ReadSectors(sectorAddress - kvp.Value, length, kvp.Key);
throw new ArgumentOutOfRangeException(nameof(sectorAddress), "Sector address not found");
}
public byte[] ReadSectorsTag(ulong sectorAddress, uint length, SectorTagType tag)
{
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap where sectorAddress >= kvp.Value
from track in Tracks where track.TrackSequence == kvp.Key
where sectorAddress - kvp.Value <
track.TrackEndSector - track.TrackStartSector + 1 select kvp)
2017-12-21 16:33:16 +00:00
return ReadSectorsTag(sectorAddress - kvp.Value, length, kvp.Key, tag);
throw new ArgumentOutOfRangeException(nameof(sectorAddress), "Sector address not found");
}
public byte[] ReadSectors(ulong sectorAddress, uint length, uint track)
{
2020-02-28 00:19:50 +00:00
var aaruTrack = new Track
{
TrackSequence = 0
};
foreach(Track bwTrack in Tracks.Where(bwTrack => bwTrack.TrackSequence == track))
2017-12-21 16:33:16 +00:00
{
2020-02-28 00:19:50 +00:00
aaruTrack = bwTrack;
break;
}
2020-06-21 14:49:25 +01:00
if(aaruTrack is null)
throw new ArgumentOutOfRangeException(nameof(track), "Track does not exist in disc image");
if(length + sectorAddress > aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1)
2017-12-19 20:33:03 +00:00
throw new ArgumentOutOfRangeException(nameof(length),
$"Requested more sectors ({length + sectorAddress}) than present in track ({aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1}), won't cross tracks");
uint sectorOffset;
uint sectorSize;
uint sectorSkip;
bool mode2 = false;
2020-02-28 00:19:50 +00:00
switch(aaruTrack.TrackType)
{
case CommonTypes.Enums.TrackType.CdMode1:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 16;
sectorSize = 2048;
sectorSkip = 288;
2017-12-19 20:33:03 +00:00
break;
}
case CommonTypes.Enums.TrackType.CdMode2Formless:
2017-12-19 20:33:03 +00:00
{
mode2 = true;
sectorOffset = 0;
sectorSize = 2352;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
case CommonTypes.Enums.TrackType.Audio:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 0;
sectorSize = 2352;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
case CommonTypes.Enums.TrackType.Data:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 0;
sectorSize = 2048;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
default: throw new FeatureSupportedButNotImplementedImageException("Unsupported track type");
}
byte[] buffer = new byte[sectorSize * length];
2020-07-20 21:11:32 +01:00
_imageStream = aaruTrack.TrackFilter.GetDataForkStream();
var br = new BinaryReader(_imageStream);
br.BaseStream.
2020-02-28 00:19:50 +00:00
Seek((long)aaruTrack.TrackFileOffset + (long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip)),
2017-12-19 20:33:03 +00:00
SeekOrigin.Begin);
if(mode2)
{
var mode2Ms = new MemoryStream((int)(sectorSize * length));
buffer = br.ReadBytes((int)(sectorSize * length));
for(int i = 0; i < length; i++)
{
byte[] sector = new byte[sectorSize];
Array.Copy(buffer, sectorSize * 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));
* TODO: * README.md: * DiscImageChef.sln: * DiscImageChef/Commands/Ls.cs: * DiscImageChef.Filters/GZip.cs: * DiscImageChef.DiscImages/BLU.cs: * DiscImageChef.DiscImages/VHD.cs: * DiscImageChef.DiscImages/VDI.cs: * DiscImageChef.DiscImages/QED.cs: * DiscImageChef.DiscImages/DIM.cs: * DiscImageChef.DiscImages/GDI.cs: * DiscImageChef.Filters/Filter.cs: * DiscImageChef/Commands/Decode.cs: * DiscImageChef.DiscImages/QCOW.cs: * DiscImageChef.Filters/Filters.cs: * DiscImageChef/Core/Statistics.cs: * DiscImageChef.DiscImages/VHDX.cs: * DiscImageChef.DiscImages/Nero.cs: * DiscImageChef/Commands/Verify.cs: * DiscImageChef.DiscImages/UDIF.cs: * DiscImageChef/Commands/Compare.cs: * DiscImageChef/Commands/Analyze.cs: * DiscImageChef.DiscImages/QCOW2.cs: * DiscImageChef/Commands/Entropy.cs: * DiscImageChef/Commands/Formats.cs: * DiscImageChef/Commands/PrintHex.cs: * DiscImageChef.DiscImages/VMware.cs: * DiscImageChef.Settings/Settings.cs: * DiscImageChef/DetectImageFormat.cs: * DiscImageChef/DiscImageChef.csproj: * DiscImageChef.DiscImages/CDRDAO.cs: * DiscImageChef.DiscImages/CPCDSK.cs: * DiscImageChef/Commands/Checksum.cs: * DiscImageChef.DiscImages/CopyQM.cs: * DiscImageChef.DiscImages/CDRWin.cs: * DiscImageChef/Commands/Configure.cs: * DiscImageChef/Commands/DumpMedia.cs: * DiscImageChef/Commands/Statistics.cs: * DiscImageChef.Filters/ZZZNoFilter.cs: * DiscImageChef.DiscImages/TeleDisk.cs: * DiscImageChef.DiscImages/Apple2MG.cs: * DiscImageChef.Filters/OffsetStream.cs: * DiscImageChef.DiscImages/Parallels.cs: * DiscImageChef/Commands/ExtractFiles.cs: * DiscImageChef.DiscImages/DiskCopy42.cs: * DiscImageChef.DiscImages/Alcohol120.cs: * DiscImageChef.DiscImages/ZZZRawImage.cs: * DiscImageChef/Commands/CreateSidecar.cs: * DiscImageChef.DiscImages/ImagePlugin.cs: * DiscImageChef.DiscImages/BlindWrite5.cs: * DiscImageChef.DiscImages/BlindWrite4.cs: * DiscImageChef.Filters/ForcedSeekStream.cs: * DiscImageChef.Filters/Properties/AssemblyInfo.cs: * DiscImageChef.Filters/DiscImageChef.Filters.csproj: * DiscImageChef.DiscImages/DiscImageChef.DiscImages.csproj: Added filters. * DiscImageChef.Filesystems/CPM/Info.cs: Do not throw identification exceptions. * DiscImageChef/Plugins.cs: Sorted plugins lists.
2016-09-05 17:37:31 +01:00
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);
try
{
Array.Copy(sector, 0, buffer, i * sectorSize, sectorSize);
}
catch(Exception e)
{
System.Console.WriteLine(e);
throw;
}
}
return buffer;
}
public byte[] ReadSectorsTag(ulong sectorAddress, uint length, uint track, SectorTagType tag)
{
2020-02-28 00:19:50 +00:00
var aaruTrack = new Track
{
TrackSequence = 0
};
foreach(Track bwTrack in Tracks.Where(bwTrack => bwTrack.TrackSequence == track))
2017-12-21 16:33:16 +00:00
{
2020-02-28 00:19:50 +00:00
aaruTrack = bwTrack;
break;
}
2020-06-21 14:49:25 +01:00
if(aaruTrack is null)
throw new ArgumentOutOfRangeException(nameof(track), "Track does not exist in disc image");
if(length + sectorAddress > aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1)
2017-12-19 20:33:03 +00:00
throw new ArgumentOutOfRangeException(nameof(length),
$"Requested more sectors ({length + sectorAddress}) than present in track ({aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1}), won't cross tracks");
uint sectorOffset;
uint sectorSize;
uint sectorSkip;
if(aaruTrack.TrackType == CommonTypes.Enums.TrackType.Data)
throw new ArgumentException("Unsupported tag requested", nameof(tag));
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: break;
case SectorTagType.CdTrackFlags:
2020-07-20 21:11:32 +01:00
if(_trackFlags.TryGetValue((uint)sectorAddress, out byte flag))
return new[]
{
flag
};
2017-12-19 20:33:03 +00:00
2018-06-22 08:08:38 +01:00
throw new ArgumentException("Unsupported tag requested", nameof(tag));
2017-12-19 20:33:03 +00:00
default: throw new ArgumentException("Unsupported tag requested", nameof(tag));
}
2020-02-28 00:19:50 +00:00
switch(aaruTrack.TrackType)
{
case CommonTypes.Enums.TrackType.CdMode1:
switch(tag)
{
case SectorTagType.CdSectorSync:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 0;
sectorSize = 12;
sectorSkip = 2340;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorHeader:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 12;
sectorSize = 4;
sectorSkip = 2336;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorSubHeader:
throw new ArgumentException("Unsupported tag requested for this track", nameof(tag));
case SectorTagType.CdSectorEcc:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 2076;
sectorSize = 276;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorEccP:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 2076;
sectorSize = 172;
sectorSkip = 104;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorEccQ:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 2248;
sectorSize = 104;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorEdc:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 2064;
sectorSize = 4;
sectorSkip = 284;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorSubchannel:
{
sectorOffset = 0;
sectorSize = 96;
sectorSkip = 0;
break;
}
2017-12-19 20:33:03 +00:00
default: throw new ArgumentException("Unsupported tag requested", nameof(tag));
}
2017-12-19 20:33:03 +00:00
break;
case CommonTypes.Enums.TrackType.CdMode2Formless:
2017-12-19 20:33:03 +00:00
{
switch(tag)
{
case SectorTagType.CdSectorSync:
case SectorTagType.CdSectorHeader:
case SectorTagType.CdSectorEcc:
case SectorTagType.CdSectorEccP:
case SectorTagType.CdSectorEccQ:
2017-12-19 20:33:03 +00:00
throw new ArgumentException("Unsupported tag requested for this track", nameof(tag));
case SectorTagType.CdSectorSubHeader:
{
sectorOffset = 0;
sectorSize = 8;
sectorSkip = 2328;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorEdc:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 2332;
sectorSize = 4;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
case SectorTagType.CdSectorSubchannel:
{
sectorOffset = 0;
sectorSize = 96;
sectorSkip = 0;
break;
}
2017-12-19 20:33:03 +00:00
default: throw new ArgumentException("Unsupported tag requested", nameof(tag));
}
2017-12-19 20:33:03 +00:00
break;
}
case CommonTypes.Enums.TrackType.Audio:
2017-12-19 20:33:03 +00:00
{
switch(tag)
{
case SectorTagType.CdSectorSubchannel:
{
sectorOffset = 0;
sectorSize = 96;
sectorSkip = 0;
break;
}
2017-12-19 20:33:03 +00:00
default: throw new ArgumentException("Unsupported tag requested", nameof(tag));
}
break;
2017-12-19 20:33:03 +00:00
}
default: throw new FeatureSupportedButNotImplementedImageException("Unsupported track type");
}
byte[] buffer = new byte[sectorSize * length];
2020-07-20 21:11:32 +01:00
_imageStream = tag == SectorTagType.CdSectorSubchannel ? aaruTrack.TrackSubchannelFilter.GetDataForkStream()
: aaruTrack.TrackFilter.GetDataForkStream();
2020-07-20 21:11:32 +01:00
var br = new BinaryReader(_imageStream);
br.BaseStream.
Seek((long)(tag == SectorTagType.CdSectorSubchannel ? aaruTrack.TrackSubchannelOffset : aaruTrack.TrackFileOffset) + (long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip)),
2017-12-19 20:33:03 +00:00
SeekOrigin.Begin);
if(sectorOffset == 0 &&
sectorSkip == 0)
buffer = br.ReadBytes((int)(sectorSize * length));
* TODO: * README.md: * DiscImageChef.sln: * DiscImageChef/Commands/Ls.cs: * DiscImageChef.Filters/GZip.cs: * DiscImageChef.DiscImages/BLU.cs: * DiscImageChef.DiscImages/VHD.cs: * DiscImageChef.DiscImages/VDI.cs: * DiscImageChef.DiscImages/QED.cs: * DiscImageChef.DiscImages/DIM.cs: * DiscImageChef.DiscImages/GDI.cs: * DiscImageChef.Filters/Filter.cs: * DiscImageChef/Commands/Decode.cs: * DiscImageChef.DiscImages/QCOW.cs: * DiscImageChef.Filters/Filters.cs: * DiscImageChef/Core/Statistics.cs: * DiscImageChef.DiscImages/VHDX.cs: * DiscImageChef.DiscImages/Nero.cs: * DiscImageChef/Commands/Verify.cs: * DiscImageChef.DiscImages/UDIF.cs: * DiscImageChef/Commands/Compare.cs: * DiscImageChef/Commands/Analyze.cs: * DiscImageChef.DiscImages/QCOW2.cs: * DiscImageChef/Commands/Entropy.cs: * DiscImageChef/Commands/Formats.cs: * DiscImageChef/Commands/PrintHex.cs: * DiscImageChef.DiscImages/VMware.cs: * DiscImageChef.Settings/Settings.cs: * DiscImageChef/DetectImageFormat.cs: * DiscImageChef/DiscImageChef.csproj: * DiscImageChef.DiscImages/CDRDAO.cs: * DiscImageChef.DiscImages/CPCDSK.cs: * DiscImageChef/Commands/Checksum.cs: * DiscImageChef.DiscImages/CopyQM.cs: * DiscImageChef.DiscImages/CDRWin.cs: * DiscImageChef/Commands/Configure.cs: * DiscImageChef/Commands/DumpMedia.cs: * DiscImageChef/Commands/Statistics.cs: * DiscImageChef.Filters/ZZZNoFilter.cs: * DiscImageChef.DiscImages/TeleDisk.cs: * DiscImageChef.DiscImages/Apple2MG.cs: * DiscImageChef.Filters/OffsetStream.cs: * DiscImageChef.DiscImages/Parallels.cs: * DiscImageChef/Commands/ExtractFiles.cs: * DiscImageChef.DiscImages/DiskCopy42.cs: * DiscImageChef.DiscImages/Alcohol120.cs: * DiscImageChef.DiscImages/ZZZRawImage.cs: * DiscImageChef/Commands/CreateSidecar.cs: * DiscImageChef.DiscImages/ImagePlugin.cs: * DiscImageChef.DiscImages/BlindWrite5.cs: * DiscImageChef.DiscImages/BlindWrite4.cs: * DiscImageChef.Filters/ForcedSeekStream.cs: * DiscImageChef.Filters/Properties/AssemblyInfo.cs: * DiscImageChef.Filters/DiscImageChef.Filters.csproj: * DiscImageChef.DiscImages/DiscImageChef.DiscImages.csproj: Added filters. * DiscImageChef.Filesystems/CPM/Info.cs: Do not throw identification exceptions. * DiscImageChef/Plugins.cs: Sorted plugins lists.
2016-09-05 17:37:31 +01:00
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 tag == SectorTagType.CdSectorSubchannel ? Subchannel.Interleave(buffer) : buffer;
}
2018-12-31 13:17:27 +00:00
public byte[] ReadSectorLong(ulong sectorAddress) => ReadSectorsLong(sectorAddress, 1);
2018-12-31 13:17:27 +00:00
public byte[] ReadSectorLong(ulong sectorAddress, uint track) => ReadSectorsLong(sectorAddress, 1, track);
public byte[] ReadSectorsLong(ulong sectorAddress, uint length)
{
foreach(KeyValuePair<uint, ulong> kvp in from kvp in _offsetMap where sectorAddress >= kvp.Value
from track in Tracks where track.TrackSequence == kvp.Key
where sectorAddress - kvp.Value <
track.TrackEndSector - track.TrackStartSector + 1 select kvp)
2017-12-21 16:33:16 +00:00
return ReadSectorsLong(sectorAddress - kvp.Value, length, kvp.Key);
throw new ArgumentOutOfRangeException(nameof(sectorAddress), "Sector address not found");
}
public byte[] ReadSectorsLong(ulong sectorAddress, uint length, uint track)
{
2020-02-28 00:19:50 +00:00
var aaruTrack = new Track
{
TrackSequence = 0
};
foreach(Track bwTrack in Tracks.Where(bwTrack => bwTrack.TrackSequence == track))
2017-12-21 16:33:16 +00:00
{
2020-02-28 00:19:50 +00:00
aaruTrack = bwTrack;
break;
}
2020-06-21 14:49:25 +01:00
if(aaruTrack is null)
throw new ArgumentOutOfRangeException(nameof(track), "Track does not exist in disc image");
if(length + sectorAddress > aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1)
2017-12-19 20:33:03 +00:00
throw new ArgumentOutOfRangeException(nameof(length),
$"Requested more sectors ({length + sectorAddress}) than present in track ({aaruTrack.TrackEndSector - aaruTrack.TrackStartSector + 1}), won't cross tracks");
uint sectorOffset;
uint sectorSize;
uint sectorSkip;
2020-02-28 00:19:50 +00:00
switch(aaruTrack.TrackType)
{
case CommonTypes.Enums.TrackType.Audio:
case CommonTypes.Enums.TrackType.CdMode1:
case CommonTypes.Enums.TrackType.CdMode2Formless:
case CommonTypes.Enums.TrackType.Data:
2017-12-19 20:33:03 +00:00
{
sectorOffset = 0;
2020-02-28 00:19:50 +00:00
sectorSize = (uint)aaruTrack.TrackRawBytesPerSector;
sectorSkip = 0;
2017-12-19 20:33:03 +00:00
break;
}
default: throw new FeatureSupportedButNotImplementedImageException("Unsupported track type");
}
2020-07-20 21:11:32 +01:00
_imageStream = aaruTrack.TrackFilter.GetDataForkStream();
var br = new BinaryReader(_imageStream);
br.BaseStream.
2020-02-28 00:19:50 +00:00
Seek((long)aaruTrack.TrackFileOffset + (long)(sectorAddress * (sectorOffset + sectorSize + sectorSkip)),
2017-12-19 20:33:03 +00:00
SeekOrigin.Begin);
byte[] buffer = br.ReadBytes((int)(sectorSize * length));
return buffer;
}
public List<Track> GetSessionTracks(Session session)
{
if(Sessions.Contains(session))
return GetSessionTracks(session.SessionSequence);
2017-12-19 20:33:03 +00:00
throw new ImageNotSupportedException("Session does not exist in disc image");
}
public List<Track> GetSessionTracks(ushort session) =>
Tracks.Where(track => track.TrackSession == session).ToList();
}
2017-12-19 20:33:03 +00:00
}