// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : Hybrid.cs // Author(s) : Natalia Portillo // // Component : Device structures decoders. // // --[ Description ] ---------------------------------------------------------- // // Decodes SCSI MMC hybrid structures. // // --[ 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-2023 Natalia Portillo // ****************************************************************************/ using System.Diagnostics.CodeAnalysis; using System.Text; using Aaru.Helpers; namespace Aaru.Decoders.SCSI.MMC; // Information from the following standards: // ANSI X3.304-1997 // T10/1048-D revision 9.0 // T10/1048-D revision 10a // T10/1228-D revision 7.0c // T10/1228-D revision 11a // T10/1363-D revision 10g // T10/1545-D revision 1d // T10/1545-D revision 5 // T10/1545-D revision 5a // T10/1675-D revision 2c // T10/1675-D revision 4 // T10/1836-D revision 2g [SuppressMessage("ReSharper", "InconsistentNaming")] [SuppressMessage("ReSharper", "MemberCanBeInternal")] [SuppressMessage("ReSharper", "MemberCanBePrivate.Global")] [SuppressMessage("ReSharper", "NotAccessedField.Global")] [SuppressMessage("ReSharper", "UnusedMember.Global")] [SuppressMessage("ReSharper", "UnusedType.Global")] public static class Hybrid { public static RecognizedFormatLayers? DecodeFormatLayers(byte[] FormatLayersResponse) { if(FormatLayersResponse == null) return null; if(FormatLayersResponse.Length < 8) return null; var decoded = new RecognizedFormatLayers { DataLength = BigEndianBitConverter.ToUInt16(FormatLayersResponse, 0), Reserved1 = FormatLayersResponse[2], Reserved2 = FormatLayersResponse[3], NumberOfLayers = FormatLayersResponse[4], Reserved3 = (byte)((FormatLayersResponse[5] & 0xC0) >> 6), DefaultFormatLayer = (byte)((FormatLayersResponse[5] & 0x30) >> 4), Reserved4 = (byte)((FormatLayersResponse[5] & 0x0C) >> 2), OnlineFormatLayer = (byte)(FormatLayersResponse[5] & 0x03), FormatLayers = new ushort[(FormatLayersResponse.Length - 6) / 2] }; for(var i = 0; i < (FormatLayersResponse.Length - 6) / 2; i++) decoded.FormatLayers[i] = BigEndianBitConverter.ToUInt16(FormatLayersResponse, i * 2 + 6); return decoded; } public static string PrettifyFormatLayers(RecognizedFormatLayers? FormatLayersResponse) { if(FormatLayersResponse == null) return null; RecognizedFormatLayers response = FormatLayersResponse.Value; var sb = new StringBuilder(); sb.AppendFormat(Localization._0_format_layers_recognized, response.NumberOfLayers); for(var i = 0; i < response.FormatLayers.Length; i++) { switch(response.FormatLayers[i]) { case (ushort)FormatLayerTypeCodes.BDLayer: { sb.AppendFormat(Localization.Layer_0_is_of_type_Blu_ray, i).AppendLine(); if(response.DefaultFormatLayer == i) sb.AppendLine(Localization.This_is_the_default_layer); if(response.OnlineFormatLayer == i) sb.AppendLine(Localization.This_is_the_layer_actually_in_use); break; } case (ushort)FormatLayerTypeCodes.CDLayer: { sb.AppendFormat(Localization.Layer_0_is_of_type_CD, i).AppendLine(); if(response.DefaultFormatLayer == i) sb.AppendLine(Localization.This_is_the_default_layer); if(response.OnlineFormatLayer == i) sb.AppendLine(Localization.This_is_the_layer_actually_in_use); break; } case (ushort)FormatLayerTypeCodes.DVDLayer: { sb.AppendFormat(Localization.Layer_0_is_of_type_DVD, i).AppendLine(); if(response.DefaultFormatLayer == i) sb.AppendLine(Localization.This_is_the_default_layer); if(response.OnlineFormatLayer == i) sb.AppendLine(Localization.This_is_the_layer_actually_in_use); break; } case (ushort)FormatLayerTypeCodes.HDDVDLayer: { sb.AppendFormat(Localization.Layer_0_is_of_type_HD_DVD, i).AppendLine(); if(response.DefaultFormatLayer == i) sb.AppendLine(Localization.This_is_the_default_layer); if(response.OnlineFormatLayer == i) sb.AppendLine(Localization.This_is_the_layer_actually_in_use); break; } default: { sb.AppendFormat(Localization.Layer_0_is_of_unknown_type_1, i, response.FormatLayers[i]). AppendLine(); if(response.DefaultFormatLayer == i) sb.AppendLine(Localization.This_is_the_default_layer); if(response.OnlineFormatLayer == i) sb.AppendLine(Localization.This_is_the_layer_actually_in_use); break; } } } return sb.ToString(); } public static string PrettifyFormatLayers(byte[] FormatLayersResponse) { RecognizedFormatLayers? decoded = DecodeFormatLayers(FormatLayersResponse); return PrettifyFormatLayers(decoded); } #region Nested type: RecognizedFormatLayers public struct RecognizedFormatLayers { /// Bytes 0 to 1 Data Length public ushort DataLength; /// Byte 2 Reserved public byte Reserved1; /// Byte 3 Reserved public byte Reserved2; /// Byte 4 Number of format layers in hybrid disc identified by drive public byte NumberOfLayers; /// Byte 5, bits 7 to 6 Reserved public byte Reserved3; /// Byte 5, bits 5 to 4 Layer no. used when disc is inserted public byte DefaultFormatLayer; /// Byte 5, bits 3 to 2 Reserved public byte Reserved4; /// Byte 5, bits 1 to 0 Layer no. currently in use public byte OnlineFormatLayer; /// Bytes 6 to end Recognized format layers public ushort[] FormatLayers; } #endregion }