// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : 11.cs // Author(s) : Natalia Portillo // // Component : Device structures decoders. // // --[ Description ] ---------------------------------------------------------- // // Decodes SCSI MODE PAGE 11h: Medium partition page (1). // // --[ 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; using System.Diagnostics.CodeAnalysis; using System.Text; namespace Aaru.Decoders.SCSI; [SuppressMessage("ReSharper", "InconsistentNaming")] [SuppressMessage("ReSharper", "MemberCanBeInternal")] [SuppressMessage("ReSharper", "MemberCanBePrivate.Global")] public static partial class Modes { #region Mode Page 0x11: Medium partition page (1) public enum PartitionSizeUnitOfMeasures : byte { /// Partition size is measures in bytes Bytes = 0, /// Partition size is measures in Kilobytes Kilobytes = 1, /// Partition size is measures in Megabytes Megabytes = 2, /// Partition size is 10eUNITS bytes Exponential = 3 } public enum MediumFormatRecognitionValues : byte { /// Logical unit is incapable of format or partition recognition Incapable = 0, /// Logical unit is capable of format recognition only FormatCapable = 1, /// Logical unit is capable of partition recognition only PartitionCapable = 2, /// Logical unit is capable of both format and partition recognition Capable = 3 } /// Medium partition page(1) Page code 0x11 public struct ModePage_11 { /// Parameters can be saved public bool PS; /// Maximum number of additional partitions supported public byte MaxAdditionalPartitions; /// Number of additional partitions to be defined for a volume public byte AdditionalPartitionsDefined; /// Device defines partitions based on its fixed definition public bool FDP; /// Device should divide medium according to the additional partitions defined field using sizes defined by device public bool SDP; /// Initiator defines number and size of partitions public bool IDP; /// Defines the unit on which the partition sizes are defined public PartitionSizeUnitOfMeasures PSUM; public bool POFM; public bool CLEAR; public bool ADDP; /// Defines the capabilities for the unit to recognize media partitions and format public MediumFormatRecognitionValues MediumFormatRecognition; public byte PartitionUnits; /// Array of partition sizes in units defined above public ushort[] PartitionSizes; } public static ModePage_11? DecodeModePage_11(byte[] pageResponse) { if((pageResponse?[0] & 0x40) == 0x40) return null; if((pageResponse?[0] & 0x3F) != 0x11) return null; if(pageResponse[1] + 2 != pageResponse.Length) return null; if(pageResponse.Length < 8) return null; var decoded = new ModePage_11(); decoded.PS |= (pageResponse[0] & 0x80) == 0x80; decoded.MaxAdditionalPartitions = pageResponse[2]; decoded.AdditionalPartitionsDefined = pageResponse[3]; decoded.FDP |= (pageResponse[4] & 0x80) == 0x80; decoded.SDP |= (pageResponse[4] & 0x40) == 0x40; decoded.IDP |= (pageResponse[4] & 0x20) == 0x20; decoded.PSUM = (PartitionSizeUnitOfMeasures)((pageResponse[4] & 0x18) >> 3); decoded.POFM |= (pageResponse[4] & 0x04) == 0x04; decoded.CLEAR |= (pageResponse[4] & 0x02) == 0x02; decoded.ADDP |= (pageResponse[4] & 0x01) == 0x01; decoded.PartitionUnits = (byte)(pageResponse[6] & 0x0F); decoded.MediumFormatRecognition = (MediumFormatRecognitionValues)pageResponse[5]; decoded.PartitionSizes = new ushort[(pageResponse.Length - 8) / 2]; for(var i = 8; i < pageResponse.Length; i += 2) { decoded.PartitionSizes[(i - 8) / 2] = (ushort)(pageResponse[i] << 8); decoded.PartitionSizes[(i - 8) / 2] += pageResponse[i + 1]; } return decoded; } public static string PrettifyModePage_11(byte[] pageResponse) => PrettifyModePage_11(DecodeModePage_11(pageResponse)); public static string PrettifyModePage_11(ModePage_11? modePage) { if(!modePage.HasValue) return null; ModePage_11 page = modePage.Value; var sb = new StringBuilder(); sb.AppendLine(Localization.SCSI_medium_partition_page); if(page.PS) sb.AppendLine("\t" + Localization.Parameters_can_be_saved); sb.AppendFormat("\t" + Localization._0_maximum_additional_partitions, page.MaxAdditionalPartitions). AppendLine(); sb.AppendFormat("\t" + Localization._0_additional_partitions_defined, page.AdditionalPartitionsDefined). AppendLine(); if(page.FDP) sb.AppendLine("\t" + Localization.Partitions_are_fixed_under_device_definitions); if(page.SDP) { sb.AppendLine("\t" + Localization.Number_of_partitions_can_be_defined_but_their_size_is_defined_by_the_device); } if(page.IDP) sb.AppendLine("\t" + Localization.Number_and_size_of_partitions_can_be_manually_defined); if(page.POFM) { sb.AppendLine("\t" + Localization. Partition_parameters_will_not_be_applied_until_a_FORMAT_MEDIUM_command_is_received); } switch(page.CLEAR) { case false when !page.ADDP: sb.AppendLine("\t" + Localization.Device_may_erase_any_or_all_partitions_on_MODE_SELECT_for_partitioning); break; case true when !page.ADDP: sb.AppendLine("\t" + Localization.Device_shall_erase_all_partitions_on_MODE_SELECT_for_partitioning); break; case false: sb.AppendLine("\t" + Localization.Device_shall_not_erase_any_partition_on_MODE_SELECT_for_partitioning); break; default: sb.AppendLine("\t" + Localization. Device_shall_erase_all_partitions_differing_on_size_on_MODE_SELECT_for_partitioning); break; } string measure; switch(page.PSUM) { case PartitionSizeUnitOfMeasures.Bytes: sb.AppendLine("\t" + Localization.Partitions_are_defined_in_bytes); measure = Localization.bytes; break; case PartitionSizeUnitOfMeasures.Kilobytes: sb.AppendLine("\t" + Localization.Partitions_are_defined_in_kilobytes); measure = Localization.kilobytes; break; case PartitionSizeUnitOfMeasures.Megabytes: sb.AppendLine("\t" + Localization.Partitions_are_defined_in_megabytes); measure = Localization.megabytes; break; case PartitionSizeUnitOfMeasures.Exponential: sb.AppendFormat("\t" + Localization.Partitions_are_defined_in_units_of_0_bytes, Math.Pow(10, page.PartitionUnits)). AppendLine(); measure = string.Format(Localization.units_of_0_bytes, Math.Pow(10, page.PartitionUnits)); break; default: sb.AppendFormat("\t" + Localization.Unknown_partition_size_unit_code_0, (byte)page.PSUM).AppendLine(); measure = Localization.units; break; } switch(page.MediumFormatRecognition) { case MediumFormatRecognitionValues.Capable: sb.AppendLine("\t" + Localization.Device_is_capable_of_recognizing_both_medium_partitions_and_format); break; case MediumFormatRecognitionValues.FormatCapable: sb.AppendLine("\t" + Localization.Device_is_capable_of_recognizing_medium_format); break; case MediumFormatRecognitionValues.PartitionCapable: sb.AppendLine("\t" + Localization.Device_is_capable_of_recognizing_medium_partitions); break; case MediumFormatRecognitionValues.Incapable: sb.AppendLine("\t" + Localization.Device_is_not_capable_of_recognizing_neither_medium_partitions_nor_format); break; default: sb.AppendFormat("\t" + Localization.Unknown_medium_recognition_code_0, (byte)page.MediumFormatRecognition). AppendLine(); break; } sb.AppendFormat("\t" + Localization.Medium_has_defined_0_partitions, page.PartitionSizes.Length).AppendLine(); for(var i = 0; i < page.PartitionSizes.Length; i++) { if(page.PartitionSizes[i] == 0) { if(page.PartitionSizes.Length == 1) sb.AppendLine("\t" + Localization.Device_recognizes_one_single_partition_spanning_whole_medium); else sb.AppendFormat("\t" + Localization.Partition_0_runs_for_rest_of_medium, i).AppendLine(); } else { sb.AppendFormat("\t" + Localization.Partition_0_is_1_2_long, i, page.PartitionSizes[i], measure). AppendLine(); } } return sb.ToString(); } #endregion Mode Page 0x11: Medium partition page (1) }