mirror of
https://github.com/aaru-dps/Aaru.Server.git
synced 2025-12-16 19:24:27 +00:00
* DiscImageChef.Decoders/SCSI/Modes.cs:
Check for vendor pages not following page format (even if they must). * DiscImageChef.Devices/Device/Constructor.cs: Some devices (at least smsc usb-floppy) crash and reset when receiving ata over the ATA PASS-THROUGH scsi command. This will check for SCSI compliance first giving devices time to reset. * DiscImageChef.Devices/Device/ScsiCommands.cs: Some devices (smsc usb floppies) return the real command result size disregarding allocation length and generating a buffer overflow. * DiscImageChef.Devices/Enums.cs: Added some vendor commands for Plextor and HL-DT-ST devices. * DiscImageChef/Commands/DeviceInfo.cs: Mode sense should be written even if it can't be decoded.
This commit is contained in:
@@ -1,3 +1,9 @@
|
|||||||
|
2015-11-05 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
|
* SCSI/Modes.cs:
|
||||||
|
Check for vendor pages not following page format (even if
|
||||||
|
they must).
|
||||||
|
|
||||||
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
* DiscImageChef.Decoders.csproj:
|
* DiscImageChef.Decoders.csproj:
|
||||||
|
|||||||
@@ -5904,6 +5904,10 @@ namespace DiscImageChef.Decoders.SCSI
|
|||||||
if (isSubpage)
|
if (isSubpage)
|
||||||
{
|
{
|
||||||
pg.PageResponse = new byte[(modeResponse[offset + 2] << 8) + modeResponse[offset + 3] + 4];
|
pg.PageResponse = new byte[(modeResponse[offset + 2] << 8) + modeResponse[offset + 3] + 4];
|
||||||
|
|
||||||
|
if((pg.PageResponse.Length + offset) > modeResponse.Length)
|
||||||
|
return decoded;
|
||||||
|
|
||||||
Array.Copy(modeResponse, offset, pg.PageResponse, 0, pg.PageResponse.Length);
|
Array.Copy(modeResponse, offset, pg.PageResponse, 0, pg.PageResponse.Length);
|
||||||
pg.Page = (byte)(modeResponse[offset] & 0x3F);
|
pg.Page = (byte)(modeResponse[offset] & 0x3F);
|
||||||
pg.Subpage = modeResponse[offset + 1];
|
pg.Subpage = modeResponse[offset + 1];
|
||||||
@@ -5912,6 +5916,10 @@ namespace DiscImageChef.Decoders.SCSI
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
pg.PageResponse = new byte[modeResponse[offset + 1] + 2];
|
pg.PageResponse = new byte[modeResponse[offset + 1] + 2];
|
||||||
|
|
||||||
|
if((pg.PageResponse.Length + offset) > modeResponse.Length)
|
||||||
|
return decoded;
|
||||||
|
|
||||||
Array.Copy(modeResponse, offset, pg.PageResponse, 0, pg.PageResponse.Length);
|
Array.Copy(modeResponse, offset, pg.PageResponse, 0, pg.PageResponse.Length);
|
||||||
pg.Page = (byte)(modeResponse[offset] & 0x3F);
|
pg.Page = (byte)(modeResponse[offset] & 0x3F);
|
||||||
pg.Subpage = 0;
|
pg.Subpage = 0;
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
2015-11-05 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
|
* Device/Constructor.cs:
|
||||||
|
Some devices (at least smsc usb-floppy) crash and reset when
|
||||||
|
receiving ata over the ATA PASS-THROUGH scsi command. This
|
||||||
|
will check for SCSI compliance first giving devices time to
|
||||||
|
reset.
|
||||||
|
|
||||||
|
* Device/ScsiCommands.cs:
|
||||||
|
Some devices (smsc usb floppies) return the real command
|
||||||
|
result size disregarding allocation length and generating a
|
||||||
|
buffer overflow.
|
||||||
|
|
||||||
|
* Enums.cs:
|
||||||
|
Added some vendor commands for Plextor and HL-DT-ST devices.
|
||||||
|
|
||||||
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
* Enums.cs:
|
* Enums.cs:
|
||||||
|
|||||||
@@ -95,33 +95,29 @@ namespace DiscImageChef.Devices
|
|||||||
AtaErrorRegistersCHS errorRegisters;
|
AtaErrorRegistersCHS errorRegisters;
|
||||||
|
|
||||||
byte[] ataBuf;
|
byte[] ataBuf;
|
||||||
bool sense = AtaIdentify(out ataBuf, out errorRegisters);
|
byte[] senseBuf;
|
||||||
|
byte[] inqBuf;
|
||||||
|
|
||||||
if (!sense)
|
bool scsiSense = ScsiInquiry(out inqBuf, out senseBuf);
|
||||||
|
|
||||||
|
if (!scsiSense)
|
||||||
{
|
{
|
||||||
type = DeviceType.ATA;
|
Decoders.SCSI.Inquiry.SCSIInquiry? Inquiry = Decoders.SCSI.Inquiry.Decode(inqBuf);
|
||||||
Decoders.ATA.Identify.IdentifyDevice? ATAID = Decoders.ATA.Identify.Decode(ataBuf);
|
|
||||||
|
|
||||||
if (ATAID.HasValue)
|
type = DeviceType.SCSI;
|
||||||
|
bool sense = ScsiInquiry(out inqBuf, out senseBuf, 0x80);
|
||||||
|
if (!sense)
|
||||||
|
serial = Decoders.SCSI.EVPD.DecodePage80(inqBuf);
|
||||||
|
|
||||||
|
if (Inquiry.HasValue)
|
||||||
{
|
{
|
||||||
string[] separated = ATAID.Value.Model.Split(' ');
|
revision = StringHandlers.SpacePaddedToString(Inquiry.Value.ProductRevisionLevel);
|
||||||
|
model = StringHandlers.SpacePaddedToString(Inquiry.Value.ProductIdentification);
|
||||||
|
manufacturer = StringHandlers.SpacePaddedToString(Inquiry.Value.VendorIdentification);
|
||||||
|
|
||||||
if (separated.Length == 1)
|
scsiType = (Decoders.SCSI.PeripheralDeviceTypes)Inquiry.Value.PeripheralDeviceType;
|
||||||
model = separated[0];
|
|
||||||
else
|
|
||||||
{
|
|
||||||
manufacturer = separated[0];
|
|
||||||
model = separated[separated.Length - 1];
|
|
||||||
}
|
|
||||||
|
|
||||||
revision = ATAID.Value.FirmwareRevision;
|
|
||||||
serial = ATAID.Value.SerialNumber;
|
|
||||||
|
|
||||||
scsiType = Decoders.SCSI.PeripheralDeviceTypes.DirectAccess;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
sense = AtapiIdentify(out ataBuf, out errorRegisters);
|
sense = AtapiIdentify(out ataBuf, out errorRegisters);
|
||||||
|
|
||||||
if (!sense)
|
if (!sense)
|
||||||
@@ -132,31 +128,32 @@ namespace DiscImageChef.Devices
|
|||||||
if (ATAID.HasValue)
|
if (ATAID.HasValue)
|
||||||
serial = ATAID.Value.SerialNumber;
|
serial = ATAID.Value.SerialNumber;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
byte[] senseBuf;
|
if (scsiSense || manufacturer == "ATA")
|
||||||
byte[] inqBuf;
|
{
|
||||||
|
bool sense = AtaIdentify(out ataBuf, out errorRegisters);
|
||||||
sense = ScsiInquiry(out inqBuf, out senseBuf);
|
|
||||||
|
|
||||||
if (!sense)
|
if (!sense)
|
||||||
{
|
{
|
||||||
Decoders.SCSI.Inquiry.SCSIInquiry? Inquiry = Decoders.SCSI.Inquiry.Decode(inqBuf);
|
type = DeviceType.ATA;
|
||||||
|
Decoders.ATA.Identify.IdentifyDevice? ATAID = Decoders.ATA.Identify.Decode(ataBuf);
|
||||||
|
|
||||||
if (type != DeviceType.ATAPI)
|
if (ATAID.HasValue)
|
||||||
{
|
{
|
||||||
type = DeviceType.SCSI;
|
string[] separated = ATAID.Value.Model.Split(' ');
|
||||||
sense = ScsiInquiry(out inqBuf, out senseBuf, 0x80);
|
|
||||||
if (!sense)
|
|
||||||
serial = Decoders.SCSI.EVPD.DecodePage80(inqBuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Inquiry.HasValue)
|
if (separated.Length == 1)
|
||||||
{
|
model = separated[0];
|
||||||
revision = StringHandlers.SpacePaddedToString(Inquiry.Value.ProductRevisionLevel);
|
else
|
||||||
model = StringHandlers.SpacePaddedToString(Inquiry.Value.ProductIdentification);
|
{
|
||||||
manufacturer = StringHandlers.SpacePaddedToString(Inquiry.Value.VendorIdentification);
|
manufacturer = separated[0];
|
||||||
|
model = separated[separated.Length - 1];
|
||||||
|
}
|
||||||
|
|
||||||
scsiType = (Decoders.SCSI.PeripheralDeviceTypes)Inquiry.Value.PeripheralDeviceType;
|
revision = ATAID.Value.FirmwareRevision;
|
||||||
|
serial = ATAID.Value.SerialNumber;
|
||||||
|
|
||||||
|
scsiType = Decoders.SCSI.PeripheralDeviceTypes.DirectAccess;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -88,9 +88,9 @@ namespace DiscImageChef.Devices
|
|||||||
/// <param name="duration">Duration in milliseconds it took for the device to execute the command.</param>
|
/// <param name="duration">Duration in milliseconds it took for the device to execute the command.</param>
|
||||||
public bool ScsiInquiry(out byte[] buffer, out byte[] senseBuffer, uint timeout, out double duration)
|
public bool ScsiInquiry(out byte[] buffer, out byte[] senseBuffer, uint timeout, out double duration)
|
||||||
{
|
{
|
||||||
buffer = new byte[5];
|
buffer = new byte[36];
|
||||||
senseBuffer = new byte[32];
|
senseBuffer = new byte[32];
|
||||||
byte[] cdb = { (byte)ScsiCommands.Inquiry, 0, 0, 0, 5, 0 };
|
byte[] cdb = { (byte)ScsiCommands.Inquiry, 0, 0, 0, 36, 0 };
|
||||||
bool sense;
|
bool sense;
|
||||||
|
|
||||||
lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense);
|
lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense);
|
||||||
@@ -163,9 +163,9 @@ namespace DiscImageChef.Devices
|
|||||||
/// <param name="page">The Extended Vital Product Data</param>
|
/// <param name="page">The Extended Vital Product Data</param>
|
||||||
public bool ScsiInquiry(out byte[] buffer, out byte[] senseBuffer, byte page, uint timeout, out double duration)
|
public bool ScsiInquiry(out byte[] buffer, out byte[] senseBuffer, byte page, uint timeout, out double duration)
|
||||||
{
|
{
|
||||||
buffer = new byte[5];
|
buffer = new byte[36];
|
||||||
senseBuffer = new byte[32];
|
senseBuffer = new byte[32];
|
||||||
byte[] cdb = { (byte)ScsiCommands.Inquiry, 1, page, 0, 5, 0 };
|
byte[] cdb = { (byte)ScsiCommands.Inquiry, 1, page, 0, 36, 0 };
|
||||||
bool sense;
|
bool sense;
|
||||||
|
|
||||||
lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense);
|
lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense);
|
||||||
@@ -174,6 +174,10 @@ namespace DiscImageChef.Devices
|
|||||||
if (sense)
|
if (sense)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
// This is because INQ was returned instead of EVPD
|
||||||
|
if (buffer[1] != page)
|
||||||
|
return true;
|
||||||
|
|
||||||
byte pagesLength = (byte)(buffer[3] + 4);
|
byte pagesLength = (byte)(buffer[3] + 4);
|
||||||
|
|
||||||
cdb = new byte[] { (byte)ScsiCommands.Inquiry, 1, page, 0, pagesLength, 0 };
|
cdb = new byte[] { (byte)ScsiCommands.Inquiry, 1, page, 0, pagesLength, 0 };
|
||||||
@@ -336,7 +340,7 @@ namespace DiscImageChef.Devices
|
|||||||
{
|
{
|
||||||
senseBuffer = new byte[32];
|
senseBuffer = new byte[32];
|
||||||
byte[] cdb = new byte[10];
|
byte[] cdb = new byte[10];
|
||||||
buffer = new byte[4];
|
buffer = new byte[4096];
|
||||||
bool sense;
|
bool sense;
|
||||||
|
|
||||||
cdb[0] = (byte)ScsiCommands.ModeSense10;
|
cdb[0] = (byte)ScsiCommands.ModeSense10;
|
||||||
|
|||||||
@@ -2559,7 +2559,29 @@ namespace DiscImageChef.Devices
|
|||||||
/// Variable sized Command Description Block
|
/// Variable sized Command Description Block
|
||||||
/// SPC-4 rev. 16
|
/// SPC-4 rev. 16
|
||||||
/// </summary>
|
/// </summary>
|
||||||
VariableSizedCDB = 0x7F
|
VariableSizedCDB = 0x7F,
|
||||||
|
|
||||||
|
#region Plextor vendor commands
|
||||||
|
/// <summary>
|
||||||
|
/// Sends extended commands (like SpeedRead) to Plextor drives
|
||||||
|
/// </summary>
|
||||||
|
Plextor_Extend = 0xE9,
|
||||||
|
/// <summary>
|
||||||
|
/// Resets Plextor drives
|
||||||
|
/// </summary>
|
||||||
|
Plextor_Reset = 0xEE,
|
||||||
|
/// <summary>
|
||||||
|
/// Reads drive statistics from Plextor drives EEPROM
|
||||||
|
/// </summary>
|
||||||
|
Plextor_ReadEeprom = 0xF1,
|
||||||
|
#endregion Plextor vendor commands
|
||||||
|
|
||||||
|
#region HL-DT-ST vendor commands
|
||||||
|
/// <summary>
|
||||||
|
/// Sends debugging commands to HL-DT-ST DVD drives
|
||||||
|
/// </summary>
|
||||||
|
HlDtSt_Vendor = 0xE7
|
||||||
|
#endregion HL-DT-ST vendor commands
|
||||||
}
|
}
|
||||||
#endregion SCSI Commands
|
#endregion SCSI Commands
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,8 @@
|
|||||||
|
2015-11-05 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
|
* Commands/DeviceInfo.cs:
|
||||||
|
Mode sense should be written even if it can't be decoded.
|
||||||
|
|
||||||
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
2015-11-02 Natalia Portillo <claunia@claunia.com>
|
||||||
|
|
||||||
* Options.cs:
|
* Options.cs:
|
||||||
|
|||||||
@@ -349,16 +349,16 @@ namespace DiscImageChef.Commands
|
|||||||
decMode = Decoders.SCSI.Modes.DecodeMode6(modeBuf, devType);
|
decMode = Decoders.SCSI.Modes.DecodeMode6(modeBuf, devType);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (decMode.HasValue)
|
if (!sense)
|
||||||
{
|
{
|
||||||
if(!string.IsNullOrEmpty(options.OutputPrefix))
|
if (!string.IsNullOrEmpty(options.OutputPrefix))
|
||||||
{
|
{
|
||||||
if (!File.Exists(options.OutputPrefix + "_scsi_modesense.bin"))
|
if (!File.Exists(options.OutputPrefix + "_scsi_modesense.bin"))
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
DicConsole.DebugWriteLine("Device-Info command", "Writing SCSI MODE SENSE to {0}{1}", options.OutputPrefix, "_scsi_modesense.bin");
|
DicConsole.DebugWriteLine("Device-Info command", "Writing SCSI MODE SENSE to {0}{1}", options.OutputPrefix, "_scsi_modesense.bin");
|
||||||
outputFs = new FileStream(options.OutputPrefix +"_scsi_modesense.bin", FileMode.CreateNew);
|
outputFs = new FileStream(options.OutputPrefix + "_scsi_modesense.bin", FileMode.CreateNew);
|
||||||
outputFs.Write(modeBuf, 0, modeBuf.Length);
|
outputFs.Write(modeBuf, 0, modeBuf.Length);
|
||||||
outputFs.Close();
|
outputFs.Close();
|
||||||
}
|
}
|
||||||
@@ -370,7 +370,10 @@ namespace DiscImageChef.Commands
|
|||||||
else
|
else
|
||||||
DicConsole.ErrorWriteLine("Not overwriting file {0}{1}", options.OutputPrefix, "_scsi_modesense.bin");
|
DicConsole.ErrorWriteLine("Not overwriting file {0}{1}", options.OutputPrefix, "_scsi_modesense.bin");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (decMode.HasValue)
|
||||||
|
{
|
||||||
DicConsole.WriteLine(Decoders.SCSI.Modes.PrettifyModeHeader(decMode.Value.Header, devType));
|
DicConsole.WriteLine(Decoders.SCSI.Modes.PrettifyModeHeader(decMode.Value.Header, devType));
|
||||||
|
|
||||||
if (decMode.Value.Pages != null)
|
if (decMode.Value.Pages != null)
|
||||||
@@ -608,7 +611,7 @@ namespace DiscImageChef.Commands
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
DicConsole.DebugWriteLine("Device-Info command", "Writing SCSI MODE SENSE to {0}{1}", options.OutputPrefix, "_mmc_getconfiguration.bin");
|
DicConsole.DebugWriteLine("Device-Info command", "Writing MMC GET CONFIGURATION to {0}{1}", options.OutputPrefix, "_mmc_getconfiguration.bin");
|
||||||
outputFs = new FileStream(options.OutputPrefix +"_mmc_getconfiguration.bin", FileMode.CreateNew);
|
outputFs = new FileStream(options.OutputPrefix +"_mmc_getconfiguration.bin", FileMode.CreateNew);
|
||||||
outputFs.Write(confBuf, 0, confBuf.Length);
|
outputFs.Write(confBuf, 0, confBuf.Length);
|
||||||
outputFs.Close();
|
outputFs.Close();
|
||||||
|
|||||||
Reference in New Issue
Block a user