2015-10-13 01:45:07 +01:00
|
|
|
|
// /***************************************************************************
|
|
|
|
|
|
// The Disc Image Chef
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
//
|
|
|
|
|
|
// Filename : DeviceInfo.cs
|
|
|
|
|
|
// Version : 1.0
|
|
|
|
|
|
// Author(s) : Natalia Portillo
|
|
|
|
|
|
//
|
|
|
|
|
|
// Component : Component
|
|
|
|
|
|
//
|
|
|
|
|
|
// Revision : $Revision$
|
|
|
|
|
|
// Last change by : $Author$
|
|
|
|
|
|
// Date : $Date$
|
|
|
|
|
|
//
|
|
|
|
|
|
// --[ Description ] ----------------------------------------------------------
|
|
|
|
|
|
//
|
|
|
|
|
|
// Description
|
|
|
|
|
|
//
|
|
|
|
|
|
// --[ License ] --------------------------------------------------------------
|
|
|
|
|
|
//
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
|
// it under the terms of the GNU General Public License as
|
|
|
|
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
|
|
//
|
|
|
|
|
|
// This program 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 General Public License for more details.
|
|
|
|
|
|
//
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
//
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
// Copyright (C) 2011-2015 Claunia.com
|
|
|
|
|
|
// ****************************************************************************/
|
|
|
|
|
|
// //$Id$
|
|
|
|
|
|
using System;
|
|
|
|
|
|
using DiscImageChef.Devices;
|
|
|
|
|
|
using System.IO;
|
2015-10-18 22:04:03 +01:00
|
|
|
|
using DiscImageChef.Console;
|
2015-10-19 04:32:16 +01:00
|
|
|
|
using System.Text;
|
2015-10-13 01:45:07 +01:00
|
|
|
|
|
|
|
|
|
|
namespace DiscImageChef.Commands
|
|
|
|
|
|
{
|
|
|
|
|
|
public static class DeviceInfo
|
|
|
|
|
|
{
|
|
|
|
|
|
public static void doDeviceInfo(DeviceInfoSubOptions options)
|
|
|
|
|
|
{
|
2015-10-18 22:04:03 +01:00
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "--debug={0}", options.Debug);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "--verbose={0}", options.Verbose);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "--device={0}", options.DevicePath);
|
2015-10-13 01:45:07 +01:00
|
|
|
|
|
|
|
|
|
|
if (options.DevicePath.Length == 2 && options.DevicePath[1] == ':' &&
|
|
|
|
|
|
options.DevicePath[0] != '/' && Char.IsLetter(options.DevicePath[0]))
|
|
|
|
|
|
{
|
|
|
|
|
|
options.DevicePath = "\\\\.\\" + Char.ToUpper(options.DevicePath[0]) + ':';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Device dev = new Device(options.DevicePath);
|
|
|
|
|
|
|
|
|
|
|
|
if (dev.Error)
|
|
|
|
|
|
{
|
2015-10-18 22:04:03 +01:00
|
|
|
|
DicConsole.ErrorWriteLine("Error {0} opening device.", dev.LastError);
|
2015-10-13 01:45:07 +01:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2015-10-19 04:32:16 +01:00
|
|
|
|
bool ata = false;
|
|
|
|
|
|
bool atapi = false;
|
|
|
|
|
|
bool scsi = false;
|
|
|
|
|
|
bool scsi83 = false;
|
2015-10-13 01:45:07 +01:00
|
|
|
|
|
2015-10-19 04:32:16 +01:00
|
|
|
|
string decodedAta = null;
|
|
|
|
|
|
string decodedAtapi = null;
|
|
|
|
|
|
string decodedScsi = null;
|
|
|
|
|
|
string scsiSerial = null;
|
2015-10-13 01:45:07 +01:00
|
|
|
|
|
2015-10-19 04:32:16 +01:00
|
|
|
|
StringBuilder sb = null;
|
2015-10-15 01:46:31 +01:00
|
|
|
|
|
|
|
|
|
|
Structs.AtaErrorRegistersCHS errorRegisters;
|
|
|
|
|
|
|
|
|
|
|
|
byte[] ataBuf;
|
2015-10-19 04:32:16 +01:00
|
|
|
|
bool sense = dev.AtaIdentify(out ataBuf, out errorRegisters);
|
2015-10-15 01:46:31 +01:00
|
|
|
|
|
|
|
|
|
|
if (sense)
|
|
|
|
|
|
{
|
2015-10-19 04:32:16 +01:00
|
|
|
|
|
2015-10-15 01:46:31 +01:00
|
|
|
|
if ((errorRegisters.status & 0x01) == 0x01
|
2015-10-19 04:32:16 +01:00
|
|
|
|
&& (errorRegisters.error & 0x04) == 0x04
|
|
|
|
|
|
&& errorRegisters.cylinderHigh == 0xEB
|
|
|
|
|
|
&& errorRegisters.cylinderLow == 0x14)
|
2015-10-15 01:46:31 +01:00
|
|
|
|
{
|
|
|
|
|
|
sense = dev.AtapiIdentify(out ataBuf, out errorRegisters);
|
|
|
|
|
|
|
|
|
|
|
|
if (sense)
|
|
|
|
|
|
{
|
2015-10-18 22:04:03 +01:00
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "STATUS = 0x{0:X2}", errorRegisters.status);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "ERROR = 0x{0:X2}", errorRegisters.error);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "NSECTOR = 0x{0:X2}", errorRegisters.sectorCount);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "SECTOR = 0x{0:X2}", errorRegisters.sector);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "CYLHIGH = 0x{0:X2}", errorRegisters.cylinderHigh);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "CYLLOW = 0x{0:X2}", errorRegisters.cylinderLow);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "DEVICE = 0x{0:X2}", errorRegisters.deviceHead);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "COMMAND = 0x{0:X2}", errorRegisters.command);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "Error code = {0}", dev.LastError);
|
2015-10-15 01:46:31 +01:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2015-10-19 04:32:16 +01:00
|
|
|
|
atapi = true;
|
|
|
|
|
|
decodedAtapi = Decoders.ATA.Identify.Prettify(ataBuf);
|
2015-10-15 01:46:31 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2015-10-18 22:04:03 +01:00
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "STATUS = 0x{0:X2}", errorRegisters.status);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "ERROR = 0x{0:X2}", errorRegisters.error);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "NSECTOR = 0x{0:X2}", errorRegisters.sectorCount);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "SECTOR = 0x{0:X2}", errorRegisters.sector);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "CYLHIGH = 0x{0:X2}", errorRegisters.cylinderHigh);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "CYLLOW = 0x{0:X2}", errorRegisters.cylinderLow);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "DEVICE = 0x{0:X2}", errorRegisters.deviceHead);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "COMMAND = 0x{0:X2}", errorRegisters.command);
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "Error code = {0}", dev.LastError);
|
2015-10-15 01:46:31 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2015-10-19 04:32:16 +01:00
|
|
|
|
ata = true;
|
|
|
|
|
|
decodedAta = Decoders.ATA.Identify.Prettify(ataBuf);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!ata)
|
|
|
|
|
|
{
|
|
|
|
|
|
byte[] senseBuf;
|
|
|
|
|
|
byte[] inqBuf;
|
|
|
|
|
|
|
|
|
|
|
|
sense = dev.ScsiInquiry(out inqBuf, out senseBuf);
|
|
|
|
|
|
|
|
|
|
|
|
if (sense)
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.ErrorWriteLine("SCSI error. Sense decoding not yet implemented.");
|
|
|
|
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
|
|
FileStream senseFs = File.Open("sense.bin", FileMode.OpenOrCreate);
|
|
|
|
|
|
senseFs.Write(senseBuf, 0, senseBuf.Length);
|
|
|
|
|
|
#endif
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
scsi = true;
|
|
|
|
|
|
decodedScsi = Decoders.SCSI.Inquiry.Prettify(inqBuf);
|
|
|
|
|
|
|
|
|
|
|
|
sense = dev.ScsiInquiry(out inqBuf, out senseBuf, 0x00);
|
|
|
|
|
|
|
|
|
|
|
|
if (!sense)
|
|
|
|
|
|
{
|
|
|
|
|
|
byte[] pages = Decoders.SCSI.EVPD.DecodePage00(inqBuf);
|
|
|
|
|
|
|
|
|
|
|
|
foreach (byte page in pages)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (page >= 0x01 && page <= 0x7F)
|
|
|
|
|
|
{
|
|
|
|
|
|
sense = dev.ScsiInquiry(out inqBuf, out senseBuf, page);
|
|
|
|
|
|
if (!sense)
|
|
|
|
|
|
{
|
|
|
|
|
|
if(sb == null)
|
|
|
|
|
|
sb = new StringBuilder();
|
|
|
|
|
|
sb.AppendFormat("Page 0x{0:X2}: ", Decoders.SCSI.EVPD.DecodeASCIIPage(inqBuf)).AppendLine();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (page == 0x80)
|
|
|
|
|
|
{
|
|
|
|
|
|
sense = dev.ScsiInquiry(out inqBuf, out senseBuf, page);
|
|
|
|
|
|
if (!sense)
|
|
|
|
|
|
{
|
|
|
|
|
|
scsi83 = true;
|
|
|
|
|
|
scsiSerial = Decoders.SCSI.EVPD.DecodePage80(inqBuf);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
if(page != 0x00)
|
|
|
|
|
|
DicConsole.DebugWriteLine("Device-Info command", "Found undecoded SCSI VPD page 0x{0:X2}", page);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (atapi)
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.WriteLine(decodedAtapi);
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (scsi)
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.WriteLine("SCSI device");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(scsi)
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.WriteLine(decodedScsi);
|
|
|
|
|
|
|
|
|
|
|
|
if(scsi83)
|
|
|
|
|
|
DicConsole.WriteLine("Unit Serial Number: {0}", scsiSerial);
|
|
|
|
|
|
|
|
|
|
|
|
if(sb != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.WriteLine("ASCII VPDs:");
|
|
|
|
|
|
DicConsole.WriteLine(sb.ToString());
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
DicConsole.WriteLine("ATA device");
|
|
|
|
|
|
DicConsole.WriteLine(decodedAta);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(!ata && !atapi && !scsi)
|
|
|
|
|
|
DicConsole.ErrorWriteLine("Unknown device type, cannot get information.");
|
2015-10-13 01:45:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|