2017-05-19 20:28:49 +01:00
|
|
|
// /***************************************************************************
|
2016-01-15 07:00:43 +00:00
|
|
|
// The Disc Image Chef
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Filename : DensitySupport.cs
|
2016-07-28 18:13:49 +01:00
|
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
2016-01-15 07:00:43 +00:00
|
|
|
//
|
2016-07-28 18:13:49 +01:00
|
|
|
// Component : Device structures decoders.
|
2016-01-15 07:00:43 +00:00
|
|
|
//
|
|
|
|
|
// --[ Description ] ----------------------------------------------------------
|
|
|
|
|
//
|
2018-12-29 15:26:00 +00:00
|
|
|
// Decodes SCSI SSC density support structures.
|
2016-01-15 07:00:43 +00:00
|
|
|
//
|
|
|
|
|
// --[ License ] --------------------------------------------------------------
|
|
|
|
|
//
|
2016-07-28 18:13:49 +01:00
|
|
|
// 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
|
2016-01-15 07:00:43 +00:00
|
|
|
// License, or (at your option) any later version.
|
|
|
|
|
//
|
2016-07-28 18:13:49 +01:00
|
|
|
// 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.
|
2016-01-15 07:00:43 +00:00
|
|
|
//
|
2016-07-28 18:13:49 +01:00
|
|
|
// 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/>.
|
2016-01-15 07:00:43 +00:00
|
|
|
//
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2017-12-19 03:50:57 +00:00
|
|
|
// Copyright © 2011-2018 Natalia Portillo
|
2016-01-15 07:00:43 +00:00
|
|
|
// ****************************************************************************/
|
2016-07-28 18:13:49 +01:00
|
|
|
|
2016-01-15 07:00:43 +00:00
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2017-12-22 02:04:18 +00:00
|
|
|
using System.Diagnostics.CodeAnalysis;
|
2016-01-15 07:00:43 +00:00
|
|
|
using System.Text;
|
|
|
|
|
|
|
|
|
|
namespace DiscImageChef.Decoders.SCSI.SSC
|
|
|
|
|
{
|
2017-12-22 02:04:18 +00:00
|
|
|
[SuppressMessage("ReSharper", "InconsistentNaming")]
|
|
|
|
|
[SuppressMessage("ReSharper", "MemberCanBeInternal")]
|
|
|
|
|
[SuppressMessage("ReSharper", "MemberCanBePrivate.Global")]
|
|
|
|
|
[SuppressMessage("ReSharper", "NotAccessedField.Global")]
|
2016-01-15 07:00:43 +00:00
|
|
|
public static class DensitySupport
|
|
|
|
|
{
|
|
|
|
|
public struct DensitySupportHeader
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
public ushort length;
|
|
|
|
|
public ushort reserved;
|
2016-01-15 07:00:43 +00:00
|
|
|
public DensitySupportDescriptor[] descriptors;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct MediaTypeSupportHeader
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
public ushort length;
|
|
|
|
|
public ushort reserved;
|
2016-01-15 07:00:43 +00:00
|
|
|
public MediaTypeSupportDescriptor[] descriptors;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct DensitySupportDescriptor
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
public byte primaryCode;
|
|
|
|
|
public byte secondaryCode;
|
|
|
|
|
public bool writable;
|
|
|
|
|
public bool duplicate;
|
|
|
|
|
public bool defaultDensity;
|
|
|
|
|
public byte reserved;
|
|
|
|
|
public bool lenvalid;
|
2016-01-15 07:00:43 +00:00
|
|
|
public ushort len;
|
2018-06-22 08:08:38 +01:00
|
|
|
public uint bpmm;
|
2016-01-15 07:00:43 +00:00
|
|
|
public ushort width;
|
|
|
|
|
public ushort tracks;
|
2018-06-22 08:08:38 +01:00
|
|
|
public uint capacity;
|
2016-01-15 07:00:43 +00:00
|
|
|
public string organization;
|
|
|
|
|
public string name;
|
|
|
|
|
public string description;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct MediaTypeSupportDescriptor
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
public byte mediumType;
|
|
|
|
|
public byte reserved1;
|
2016-01-15 07:00:43 +00:00
|
|
|
public ushort len;
|
2018-06-22 08:08:38 +01:00
|
|
|
public byte numberOfCodes;
|
2016-01-15 07:00:43 +00:00
|
|
|
public byte[] densityCodes;
|
|
|
|
|
public ushort width;
|
|
|
|
|
public ushort length;
|
2018-06-22 08:08:38 +01:00
|
|
|
public byte reserved2;
|
|
|
|
|
public byte reserved3;
|
2016-01-15 07:00:43 +00:00
|
|
|
public string organization;
|
|
|
|
|
public string name;
|
|
|
|
|
public string description;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static DensitySupportHeader? DecodeDensity(byte[] response)
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response == null) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response.Length <= 56) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
ushort responseLen = (ushort)((response[0] << 8) + response[1] + 2);
|
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response.Length != responseLen) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
List<DensitySupportDescriptor> descriptors = new List<DensitySupportDescriptor>();
|
2018-06-22 08:08:38 +01:00
|
|
|
int offset = 4;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2016-04-19 02:11:47 +01:00
|
|
|
while(offset < response.Length)
|
2016-01-15 07:00:43 +00:00
|
|
|
{
|
2017-12-22 02:04:18 +00:00
|
|
|
DensitySupportDescriptor descriptor = new DensitySupportDescriptor
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
primaryCode = response[offset + 0],
|
|
|
|
|
secondaryCode = response[offset + 1],
|
|
|
|
|
writable = (response[offset + 2] & 0x80) == 0x80,
|
|
|
|
|
duplicate = (response[offset + 2] & 0x40) == 0x40,
|
2017-12-22 02:04:18 +00:00
|
|
|
defaultDensity = (response[offset + 2] & 0x20) == 0x20,
|
2018-06-22 08:08:38 +01:00
|
|
|
reserved = (byte)((response[offset + 2] & 0x1E) >> 1),
|
|
|
|
|
lenvalid =
|
|
|
|
|
(response[offset + 2] &
|
|
|
|
|
0x01) == 0x01,
|
|
|
|
|
len =
|
|
|
|
|
(ushort)((response[offset + 3] << 8) + response[offset + 4]),
|
|
|
|
|
bpmm =
|
|
|
|
|
(uint)((response[offset + 5] << 16) + (response[offset + 6] << 8) + response[offset + 7]),
|
|
|
|
|
width =
|
|
|
|
|
(ushort)((response[offset + 8] << 8) + response[offset + 9]),
|
|
|
|
|
tracks =
|
|
|
|
|
(ushort)((response[offset + 10] << 8) + response[offset + 11]),
|
2017-12-22 02:04:18 +00:00
|
|
|
capacity = (uint)((response[offset + 12] << 24) + (response[offset + 13] << 16) +
|
2018-06-22 08:08:38 +01:00
|
|
|
(response[offset + 14] << 8) + response[offset + 15])
|
2017-12-22 02:04:18 +00:00
|
|
|
};
|
|
|
|
|
byte[] tmp = new byte[8];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 16, tmp, 0, 8);
|
|
|
|
|
descriptor.organization = StringHandlers.CToString(tmp).Trim();
|
2018-06-22 08:08:38 +01:00
|
|
|
tmp = new byte[8];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 24, tmp, 0, 8);
|
|
|
|
|
descriptor.name = StringHandlers.CToString(tmp).Trim();
|
2018-06-22 08:08:38 +01:00
|
|
|
tmp = new byte[20];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 32, tmp, 0, 20);
|
|
|
|
|
descriptor.description = StringHandlers.CToString(tmp).Trim();
|
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
if(descriptor.lenvalid) offset += descriptor.len + 5;
|
2018-06-22 08:08:38 +01:00
|
|
|
else offset += 52;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
descriptors.Add(descriptor);
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 02:04:18 +00:00
|
|
|
DensitySupportHeader decoded = new DensitySupportHeader
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
length = responseLen,
|
|
|
|
|
reserved = (ushort)((response[2] << 8) + response[3] + 2),
|
2017-12-22 02:04:18 +00:00
|
|
|
descriptors = descriptors.ToArray()
|
|
|
|
|
};
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
return decoded;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string PrettifyDensity(DensitySupportHeader? density)
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
if(density == null) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
DensitySupportHeader decoded = density.Value;
|
2018-06-22 08:08:38 +01:00
|
|
|
StringBuilder sb = new StringBuilder();
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2016-04-19 02:11:47 +01:00
|
|
|
foreach(DensitySupportDescriptor descriptor in decoded.descriptors)
|
2016-01-15 07:00:43 +00:00
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
sb.AppendFormat("Density \"{0}\" defined by \"{1}\".", descriptor.name, descriptor.organization)
|
|
|
|
|
.AppendLine();
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendFormat("\tPrimary code: {0:X2}h", descriptor.primaryCode).AppendLine();
|
|
|
|
|
if(descriptor.primaryCode != descriptor.secondaryCode)
|
|
|
|
|
sb.AppendFormat("\tSecondary code: {0:X2}h", descriptor.secondaryCode).AppendLine();
|
2017-12-19 20:33:03 +00:00
|
|
|
if(descriptor.writable) sb.AppendLine("\tDrive can write this density");
|
|
|
|
|
if(descriptor.duplicate) sb.AppendLine("\tThis descriptor is duplicated");
|
|
|
|
|
if(descriptor.defaultDensity) sb.AppendLine("\tThis is the default density on the drive");
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendFormat("\tDensity has {0} bits per mm, with {1} tracks in a {2} mm width tape",
|
2017-12-23 18:31:38 +00:00
|
|
|
descriptor.bpmm, descriptor.tracks, descriptor.width / (double)10).AppendLine();
|
2016-07-28 23:08:22 +01:00
|
|
|
sb.AppendFormat("\tDensity maximum capacity is {0} megabytes", descriptor.capacity).AppendLine();
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendFormat("\tDensity description: {0}", descriptor.description).AppendLine();
|
|
|
|
|
sb.AppendLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string PrettifyDensity(byte[] response)
|
|
|
|
|
{
|
|
|
|
|
return PrettifyDensity(DecodeDensity(response));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static MediaTypeSupportHeader? DecodeMediumType(byte[] response)
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response == null) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response.Length <= 60) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
ushort responseLen = (ushort)((response[0] << 8) + response[1] + 2);
|
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
if(response.Length != responseLen) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
List<MediaTypeSupportDescriptor> descriptors = new List<MediaTypeSupportDescriptor>();
|
2018-06-22 08:08:38 +01:00
|
|
|
int offset = 4;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2016-04-19 02:11:47 +01:00
|
|
|
while(offset < response.Length)
|
2016-01-15 07:00:43 +00:00
|
|
|
{
|
2017-12-22 02:04:18 +00:00
|
|
|
MediaTypeSupportDescriptor descriptor = new MediaTypeSupportDescriptor
|
|
|
|
|
{
|
|
|
|
|
mediumType = response[offset + 0],
|
2018-06-22 08:08:38 +01:00
|
|
|
reserved1 = response[offset + 1],
|
|
|
|
|
len = (ushort)((response[offset + 2] << 8) + response[offset + 3])
|
2017-12-22 02:04:18 +00:00
|
|
|
};
|
2017-12-19 20:33:03 +00:00
|
|
|
if(descriptor.len != 52) return null;
|
|
|
|
|
|
2016-01-15 07:00:43 +00:00
|
|
|
descriptor.numberOfCodes = response[offset + 4];
|
2018-06-22 08:08:38 +01:00
|
|
|
descriptor.densityCodes = new byte[9];
|
|
|
|
|
Array.Copy(response, offset + 5, descriptor.densityCodes, 0, 9);
|
|
|
|
|
descriptor.width = (ushort)((response[offset + 14] << 8) + response[offset + 15]);
|
|
|
|
|
descriptor.length = (ushort)((response[offset + 16] << 8) + response[offset + 17]);
|
2016-01-15 07:00:43 +00:00
|
|
|
descriptor.reserved1 = response[offset + 18];
|
|
|
|
|
descriptor.reserved1 = response[offset + 19];
|
2017-12-22 02:04:18 +00:00
|
|
|
byte[] tmp = new byte[8];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 20, tmp, 0, 8);
|
|
|
|
|
descriptor.organization = StringHandlers.CToString(tmp).Trim();
|
2018-06-22 08:08:38 +01:00
|
|
|
tmp = new byte[8];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 28, tmp, 0, 8);
|
|
|
|
|
descriptor.name = StringHandlers.CToString(tmp).Trim();
|
2018-06-22 08:08:38 +01:00
|
|
|
tmp = new byte[20];
|
2016-01-15 07:00:43 +00:00
|
|
|
Array.Copy(response, offset + 36, tmp, 0, 20);
|
|
|
|
|
descriptor.description = StringHandlers.CToString(tmp).Trim();
|
|
|
|
|
|
|
|
|
|
offset += 56;
|
|
|
|
|
|
|
|
|
|
descriptors.Add(descriptor);
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 02:04:18 +00:00
|
|
|
MediaTypeSupportHeader decoded = new MediaTypeSupportHeader
|
|
|
|
|
{
|
2018-06-22 08:08:38 +01:00
|
|
|
length = responseLen,
|
|
|
|
|
reserved = (ushort)((response[2] << 8) + response[3] + 2),
|
2017-12-22 02:04:18 +00:00
|
|
|
descriptors = descriptors.ToArray()
|
|
|
|
|
};
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
return decoded;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string PrettifyMediumType(MediaTypeSupportHeader? mediumType)
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
if(mediumType == null) return null;
|
2016-01-15 07:00:43 +00:00
|
|
|
|
|
|
|
|
MediaTypeSupportHeader decoded = mediumType.Value;
|
2018-06-22 08:08:38 +01:00
|
|
|
StringBuilder sb = new StringBuilder();
|
2016-01-15 07:00:43 +00:00
|
|
|
|
2016-04-19 02:11:47 +01:00
|
|
|
foreach(MediaTypeSupportDescriptor descriptor in decoded.descriptors)
|
2016-01-15 07:00:43 +00:00
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
sb.AppendFormat("Medium type \"{0}\" defined by \"{1}\".", descriptor.name, descriptor.organization)
|
|
|
|
|
.AppendLine();
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendFormat("\tMedium type code: {0:X2}h", descriptor.mediumType).AppendLine();
|
2016-04-19 02:11:47 +01:00
|
|
|
if(descriptor.numberOfCodes > 0)
|
2016-01-15 07:00:43 +00:00
|
|
|
{
|
|
|
|
|
sb.AppendFormat("\tMedium supports following density codes:");
|
2016-04-19 02:11:47 +01:00
|
|
|
for(int i = 0; i < descriptor.numberOfCodes; i++)
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendFormat(" {0:X2}h", descriptor.densityCodes[i]);
|
2017-12-19 20:33:03 +00:00
|
|
|
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendLine();
|
|
|
|
|
}
|
2016-04-19 02:11:47 +01:00
|
|
|
|
2017-12-23 18:31:38 +00:00
|
|
|
sb.AppendFormat("\tMedium has a nominal length of {0} m in a {1} mm width tape", descriptor.length,
|
|
|
|
|
descriptor.width / (double)10).AppendLine();
|
2016-07-28 23:08:22 +01:00
|
|
|
sb.AppendFormat("\tMedium description: {0}", descriptor.description).AppendLine();
|
2016-01-15 07:00:43 +00:00
|
|
|
sb.AppendLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string PrettifyMediumType(byte[] response)
|
|
|
|
|
{
|
|
|
|
|
return PrettifyMediumType(DecodeMediumType(response));
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
}
|