// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : DeviceReport.cs // Author(s) : Natalia Portillo // // Component : Verbs. // // --[ Description ] ---------------------------------------------------------- // // Implements the 'device-report' verb. // // --[ 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 . // // ---------------------------------------------------------------------------- // Copyright © 2011-2018 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.IO; using System.Threading; using DiscImageChef.CommonTypes.Enums; using DiscImageChef.CommonTypes.Interop; using DiscImageChef.CommonTypes.Metadata; using DiscImageChef.Console; using DiscImageChef.Core; using DiscImageChef.Database; using DiscImageChef.Database.Models; using DiscImageChef.Decoders.ATA; using DiscImageChef.Decoders.SCSI; using Newtonsoft.Json; using Device = DiscImageChef.Devices.Device; namespace DiscImageChef.Commands { static class DeviceReport { internal static void DoDeviceReport(DeviceReportOptions options) { DicConsole.DebugWriteLine("Device-Report command", "--debug={0}", options.Debug); DicConsole.DebugWriteLine("Device-Report command", "--verbose={0}", options.Verbose); DicConsole.DebugWriteLine("Device-Report command", "--device={0}", options.DevicePath); if(!DetectOS.IsAdmin) { DicConsole .ErrorWriteLine("Because of the commands sent to a device, device report must be run with administrative privileges."); DicConsole.ErrorWriteLine("Not continuing."); return; } 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) { DicConsole.ErrorWriteLine("Error {0} opening device.", dev.LastError); return; } Core.Statistics.AddDevice(dev); DeviceReportV2 report = new DeviceReportV2 { Manufacturer = dev.Manufacturer, Model = dev.Model, Revision = dev.Revision, Type = dev.Type }; bool removable = false; string jsonFile; if(!string.IsNullOrWhiteSpace(dev.Manufacturer) && !string.IsNullOrWhiteSpace(dev.Revision)) jsonFile = dev.Manufacturer + "_" + dev.Model + "_" + dev.Revision + ".json"; else if(!string.IsNullOrWhiteSpace(dev.Manufacturer)) jsonFile = dev.Manufacturer + "_" + dev.Model + ".json"; else if(!string.IsNullOrWhiteSpace(dev.Revision)) jsonFile = dev.Model + "_" + dev.Revision + ".json"; else jsonFile = dev.Model + ".json"; jsonFile = jsonFile.Replace('\\', '_').Replace('/', '_').Replace('?', '_'); Core.Devices.Report.DeviceReport reporter = new Core.Devices.Report.DeviceReport(dev, options.Debug); ConsoleKeyInfo pressedKey; if(dev.IsUsb) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Is the device natively USB (in case of doubt, press Y)? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key == ConsoleKey.Y) { report.USB = reporter.UsbReport(); pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Is the media removable from the reading/writing elements? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } report.USB.RemovableMedia = pressedKey.Key == ConsoleKey.Y; removable = report.USB.RemovableMedia; } } if(dev.IsFireWire) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Is the device natively FireWire (in case of doubt, press Y)? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key != ConsoleKey.Y) { report.FireWire = reporter.FireWireReport(); pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Is the media removable from the reading/writing elements? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } report.FireWire.RemovableMedia = pressedKey.Key == ConsoleKey.Y; removable = report.FireWire.RemovableMedia; } } if(dev.IsPcmcia) report.PCMCIA = reporter.PcmciaReport(); byte[] buffer; string mediumTypeName; string mediumModel; switch(dev.Type) { case DeviceType.ATA: { DicConsole.WriteLine("Querying ATA IDENTIFY..."); dev.AtaIdentify(out buffer, out _, dev.Timeout, out _); if(!Identify.Decode(buffer).HasValue) break; report.ATA = new Ata {Identify = buffer}; if(report.ATA.IdentifyDevice == null) break; if((ushort)report.ATA.IdentifyDevice?.GeneralConfiguration == 0x848A) { report.CompactFlash = true; removable = false; } else if(!removable && report.ATA.IdentifyDevice?.GeneralConfiguration.HasFlag(Identify.GeneralConfigurationBit .Removable) == true) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Is the media removable from the reading/writing elements? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } removable = pressedKey.Key == ConsoleKey.Y; } if(removable) { DicConsole .WriteLine("Please remove any media from the device and press any key when it is out."); System.Console.ReadKey(true); DicConsole.WriteLine("Querying ATA IDENTIFY..."); dev.AtaIdentify(out buffer, out _, dev.Timeout, out _); report.ATA.Identify = buffer; List mediaTests = new List(); pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.N) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Do you have media that you can insert in the drive? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key != ConsoleKey.Y) continue; DicConsole.WriteLine("Please insert it in the drive and press any key when it is ready."); System.Console.ReadKey(true); DicConsole.Write("Please write a description of the media type and press enter: "); mediumTypeName = System.Console.ReadLine(); DicConsole.Write("Please write the media model and press enter: "); mediumModel = System.Console.ReadLine(); TestedMedia mediaTest = reporter.ReportAtaMedia(); mediaTest.MediumTypeName = mediumTypeName; mediaTest.Model = mediumModel; mediaTests.Add(mediaTest); } report.ATA.RemovableMedias = mediaTests; } else report.ATA.ReadCapabilities = reporter.ReportAta(report.ATA.IdentifyDevice.Value); break; } case DeviceType.MMC: report.MultiMediaCard = reporter.MmcSdReport(); break; case DeviceType.SecureDigital: report.SecureDigital = reporter.MmcSdReport(); break; case DeviceType.NVMe: throw new NotImplementedException("NVMe devices not yet supported."); case DeviceType.ATAPI: DicConsole.WriteLine("Querying ATAPI IDENTIFY..."); dev.AtapiIdentify(out buffer, out _, dev.Timeout, out _); if(!Identify.Decode(buffer).HasValue) return; Identify.IdentifyDevice? atapiIdNullable = Identify.Decode(buffer); report.ATAPI.Identify = buffer; goto case DeviceType.SCSI; case DeviceType.SCSI: if(!dev.IsUsb && !dev.IsFireWire && dev.IsRemovable) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Is the media removable from the reading/writing elements (flash memories ARE NOT removable)? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } removable = pressedKey.Key == ConsoleKey.Y; } if(removable) { switch(dev.ScsiType) { case PeripheralDeviceTypes.MultiMediaDevice: dev.AllowMediumRemoval(out buffer, dev.Timeout, out _); dev.EjectTray(out buffer, dev.Timeout, out _); break; case PeripheralDeviceTypes.SequentialAccess: dev.SpcAllowMediumRemoval(out buffer, dev.Timeout, out _); DicConsole.WriteLine("Asking drive to unload tape (can take a few minutes)..."); dev.Unload(out buffer, dev.Timeout, out _); break; } DicConsole .WriteLine("Please remove any media from the device and press any key when it is out."); System.Console.ReadKey(true); } report.SCSI = reporter.ReportScsiInquiry(); if(report.SCSI == null) break; report.SCSI.EVPDPages = reporter.ReportEvpdPages(); reporter.ReportScsiModes(ref report, out byte[] cdromMode); string mediumManufacturer; byte[] senseBuffer; bool sense; switch(dev.ScsiType) { case PeripheralDeviceTypes.MultiMediaDevice: { List mediaTypes = new List(); report.SCSI.MultiMediaDevice = new Mmc { ModeSense2AData = cdromMode, Features = reporter.ReportMmcFeatures() }; if(cdromMode != null) { mediaTypes.Add("CD-ROM"); mediaTypes.Add("Audio CD"); if(report.SCSI.MultiMediaDevice.ModeSense2A.ReadCDR) mediaTypes.Add("CD-R"); if(report.SCSI.MultiMediaDevice.ModeSense2A.ReadCDRW) mediaTypes.Add("CD-RW"); if(report.SCSI.MultiMediaDevice.ModeSense2A.ReadDVDROM) mediaTypes.Add("DVD-ROM"); if(report.SCSI.MultiMediaDevice.ModeSense2A.ReadDVDRAM) mediaTypes.Add("DVD-RAM"); if(report.SCSI.MultiMediaDevice.ModeSense2A.ReadDVDR) mediaTypes.Add("DVD-R"); } if(report.SCSI.MultiMediaDevice.Features != null) { if(report.SCSI.MultiMediaDevice.Features.CanReadBD || report.SCSI.MultiMediaDevice.Features.CanReadBDR || report.SCSI.MultiMediaDevice.Features.CanReadBDRE1 || report.SCSI.MultiMediaDevice.Features.CanReadBDRE2 || report.SCSI.MultiMediaDevice.Features.CanReadBDROM || report.SCSI.MultiMediaDevice.Features.CanReadOldBDR || report.SCSI.MultiMediaDevice.Features.CanReadOldBDRE || report.SCSI.MultiMediaDevice.Features.CanReadOldBDROM) { if(!mediaTypes.Contains("BD-ROM")) mediaTypes.Add("BD-ROM"); if(!mediaTypes.Contains("BD-R")) mediaTypes.Add("BD-R"); if(!mediaTypes.Contains("BD-RE")) mediaTypes.Add("BD-RE"); if(!mediaTypes.Contains("BD-R LTH")) mediaTypes.Add("BD-R LTH"); if(!mediaTypes.Contains("BD-R XL")) mediaTypes.Add("BD-R XL"); } if(report.SCSI.MultiMediaDevice.Features.CanReadCD || report.SCSI.MultiMediaDevice.Features.MultiRead) { if(!mediaTypes.Contains("CD-ROM")) mediaTypes.Add("CD-ROM"); if(!mediaTypes.Contains("Audio CD")) mediaTypes.Add("Audio CD"); if(!mediaTypes.Contains("CD-R")) mediaTypes.Add("CD-R"); if(!mediaTypes.Contains("CD-RW")) mediaTypes.Add("CD-RW"); } if(report.SCSI.MultiMediaDevice.Features.CanReadCDMRW) if(!mediaTypes.Contains("CD-MRW")) mediaTypes.Add("CD-MRW"); if(report.SCSI.MultiMediaDevice.Features.CanReadDDCD) { if(!mediaTypes.Contains("DDCD-ROM")) mediaTypes.Add("DDCD-ROM"); if(!mediaTypes.Contains("DDCD-R")) mediaTypes.Add("DDCD-R"); if(!mediaTypes.Contains("DDCD-RW")) mediaTypes.Add("DDCD-RW"); } if(report.SCSI.MultiMediaDevice.Features.CanReadDVD || report.SCSI.MultiMediaDevice.Features.DVDMultiRead || report.SCSI.MultiMediaDevice.Features.CanReadDVDPlusR || report.SCSI.MultiMediaDevice.Features.CanReadDVDPlusRDL || report.SCSI.MultiMediaDevice.Features.CanReadDVDPlusRW || report.SCSI.MultiMediaDevice.Features.CanReadDVDPlusRWDL) { if(!mediaTypes.Contains("DVD-ROM")) mediaTypes.Add("DVD-ROM"); if(!mediaTypes.Contains("DVD-R")) mediaTypes.Add("DVD-R"); if(!mediaTypes.Contains("DVD-RW")) mediaTypes.Add("DVD-RW"); if(!mediaTypes.Contains("DVD+R")) mediaTypes.Add("DVD+R"); if(!mediaTypes.Contains("DVD+RW")) mediaTypes.Add("DVD+RW"); if(!mediaTypes.Contains("DVD-R DL")) mediaTypes.Add("DVD-R DL"); if(!mediaTypes.Contains("DVD+R DL")) mediaTypes.Add("DVD+R DL"); } if(report.SCSI.MultiMediaDevice.Features.CanReadDVDPlusMRW) if(!mediaTypes.Contains("DVD+MRW")) mediaTypes.Add("DVD+MRW"); if(report.SCSI.MultiMediaDevice.Features.CanReadHDDVD || report.SCSI.MultiMediaDevice.Features.CanReadHDDVDR) { if(!mediaTypes.Contains("HD DVD-ROM")) mediaTypes.Add("HD DVD-ROM"); if(!mediaTypes.Contains("HD DVD-R")) mediaTypes.Add("HD DVD-R"); if(!mediaTypes.Contains("HD DVD-RW")) mediaTypes.Add("HD DVD-RW"); } if(report.SCSI.MultiMediaDevice.Features.CanReadHDDVDRAM) if(!mediaTypes.Contains("HD DVD-RAM")) mediaTypes.Add("HD DVD-RAM"); } // Very old CD drives do not contain mode page 2Ah neither GET CONFIGURATION, so just try all CDs on them // Also don't get confident, some drives didn't know CD-RW but are able to read them if(mediaTypes.Count == 0 || mediaTypes.Contains("CD-ROM")) { if(!mediaTypes.Contains("CD-ROM")) mediaTypes.Add("CD-ROM"); if(!mediaTypes.Contains("Audio CD")) mediaTypes.Add("Audio CD"); if(!mediaTypes.Contains("CD-R")) mediaTypes.Add("CD-R"); if(!mediaTypes.Contains("CD-RW")) mediaTypes.Add("CD-RW"); } mediaTypes.Sort(); bool tryPlextor = false, tryHldtst = false, tryPioneer = false, tryNec = false; tryPlextor |= dev.Manufacturer.ToLowerInvariant() == "plextor"; tryHldtst |= dev.Manufacturer.ToLowerInvariant() == "hl-dt-st"; tryPioneer |= dev.Manufacturer.ToLowerInvariant() == "pioneer"; tryNec |= dev.Manufacturer.ToLowerInvariant() == "nec"; if(options.Debug) { if(!tryPlextor) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Do you have want to try Plextor vendor commands? THIS IS DANGEROUS AND CAN IRREVERSIBLY DESTROY YOUR DRIVE (IF IN DOUBT PRESS 'N') (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } tryPlextor |= pressedKey.Key == ConsoleKey.Y; } if(!tryNec) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Do you have want to try NEC vendor commands? THIS IS DANGEROUS AND CAN IRREVERSIBLY DESTROY YOUR DRIVE (IF IN DOUBT PRESS 'N') (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } tryNec |= pressedKey.Key == ConsoleKey.Y; } if(!tryPioneer) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Do you have want to try Pioneer vendor commands? THIS IS DANGEROUS AND CAN IRREVERSIBLY DESTROY YOUR DRIVE (IF IN DOUBT PRESS 'N') (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } tryPioneer |= pressedKey.Key == ConsoleKey.Y; } if(!tryHldtst) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Do you have want to try HL-DT-ST (aka LG) vendor commands? THIS IS DANGEROUS AND CAN IRREVERSIBLY DESTROY YOUR DRIVE (IF IN DOUBT PRESS 'N') (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } tryHldtst |= pressedKey.Key == ConsoleKey.Y; } } List mediaTests = new List(); foreach(string mediaType in mediaTypes) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Do you have a {0} disc that you can insert in the drive? (Y/N): ", mediaType); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key != ConsoleKey.Y) continue; dev.AllowMediumRemoval(out senseBuffer, dev.Timeout, out _); dev.EjectTray(out senseBuffer, dev.Timeout, out _); DicConsole .WriteLine("Please insert it in the drive and press any key when it is ready."); System.Console.ReadKey(true); bool mediaIsRecognized = true; sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(sense) { FixedSense? decSense = Sense.DecodeFixed(senseBuffer); if(decSense.HasValue) if(decSense.Value.ASC == 0x3A) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else if(decSense.Value.ASC == 0x04 && decSense.Value.ASCQ == 0x01) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } // These should be trapped by the OS but seems in some cases they're not else if(decSense.Value.ASC == 0x28) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else mediaIsRecognized = false; else mediaIsRecognized = false; } TestedMedia mediaTest = new TestedMedia(); if(mediaIsRecognized) { mediaTest = reporter.ReportMmcMedia(mediaType, tryPlextor, tryPioneer, tryNec, tryHldtst); if(mediaTest.SupportsReadLong == true && mediaTest.LongBlockSize == mediaTest.BlockSize) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Drive supports SCSI READ LONG but I cannot find the correct size. Do you want me to try? (This can take hours) (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key == ConsoleKey.Y) { for(ushort i = (ushort)mediaTest.BlockSize;; i++) { DicConsole.Write("\rTrying to READ LONG with a size of {0} bytes...", i); sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, i, dev.Timeout, out _); if(!sense) { if(options.Debug) { FileStream bingo = new FileStream($"{mediaType}_readlong.bin", FileMode.Create); bingo.Write(buffer, 0, buffer.Length); bingo.Close(); } mediaTest.LongBlockSize = i; break; } if(i == ushort.MaxValue) break; } DicConsole.WriteLine(); } } if(options.Debug && mediaTest.SupportsReadLong == true && mediaTest.LongBlockSize != mediaTest.BlockSize) { sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, (ushort)mediaTest.LongBlockSize, dev.Timeout, out _); if(!sense) DataFile.WriteTo("SCSI Report", "readlong10", "_debug_" + dev.Model + "_" + mediaType + ".bin", "read results", buffer); } } mediaTest.MediumTypeName = mediaType; mediaTest.MediaIsRecognized = mediaIsRecognized; mediaTests.Add(mediaTest); } report.SCSI.MultiMediaDevice.TestedMedia = mediaTests; } break; case PeripheralDeviceTypes.SequentialAccess: { report.SCSI.SequentialDevice = reporter.ReportScsiSsc(); List seqTests = new List(); pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.N) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Do you have media that you can insert in the drive? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key != ConsoleKey.Y) continue; DicConsole .WriteLine("Please insert it in the drive and press any key when it is ready."); System.Console.ReadKey(true); DicConsole.Write("Please write a description of the media type and press enter: "); mediumTypeName = System.Console.ReadLine(); DicConsole.Write("Please write the media manufacturer and press enter: "); mediumManufacturer = System.Console.ReadLine(); DicConsole.Write("Please write the media model and press enter: "); mediumModel = System.Console.ReadLine(); bool mediaIsRecognized = true; dev.Load(out senseBuffer, dev.Timeout, out _); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(sense) { FixedSense? decSense = Sense.DecodeFixed(senseBuffer); if(decSense.HasValue) if(decSense.Value.ASC == 0x3A) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else if(decSense.Value.ASC == 0x04 && decSense.Value.ASCQ == 0x01) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else mediaIsRecognized = false; else mediaIsRecognized = false; } TestedSequentialMedia seqTest = new TestedSequentialMedia(); if(mediaIsRecognized) seqTest = reporter.ReportSscMedia(); seqTest.MediumTypeName = mediumTypeName; seqTest.Manufacturer = mediumManufacturer; seqTest.Model = mediumModel; seqTest.MediaIsRecognized = mediaIsRecognized; seqTests.Add(seqTest); } report.SCSI.SequentialDevice.TestedMedia = seqTests; } break; default: { if(removable) { List mediaTests = new List(); pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.N) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole.Write("Do you have media that you can insert in the drive? (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key != ConsoleKey.Y) continue; DicConsole .WriteLine("Please insert it in the drive and press any key when it is ready."); System.Console.ReadKey(true); DicConsole.Write("Please write a description of the media type and press enter: "); mediumTypeName = System.Console.ReadLine(); DicConsole.Write("Please write the media manufacturer and press enter: "); mediumManufacturer = System.Console.ReadLine(); DicConsole.Write("Please write the media model and press enter: "); mediumModel = System.Console.ReadLine(); bool mediaIsRecognized = true; sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(sense) { FixedSense? decSense = Sense.DecodeFixed(senseBuffer); if(decSense.HasValue) if(decSense.Value.ASC == 0x3A) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else if(decSense.Value.ASC == 0x04 && decSense.Value.ASCQ == 0x01) { int leftRetries = 20; while(leftRetries > 0) { DicConsole.Write("\rWaiting for drive to become ready"); Thread.Sleep(2000); sense = dev.ScsiTestUnitReady(out senseBuffer, dev.Timeout, out _); if(!sense) break; leftRetries--; } mediaIsRecognized &= !sense; } else mediaIsRecognized = false; else mediaIsRecognized = false; } TestedMedia mediaTest = new TestedMedia(); if(mediaIsRecognized) { mediaTest = reporter.ReportScsiMedia(); if(mediaTest.SupportsReadLong == true && mediaTest.LongBlockSize == mediaTest.BlockSize) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Drive supports SCSI READ LONG but I cannot find the correct size. Do you want me to try? (This can take hours) (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key == ConsoleKey.Y) { for(ushort i = (ushort)mediaTest.BlockSize;; i++) { DicConsole .Write("\rTrying to READ LONG with a size of {0} bytes...", i); sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, i, dev.Timeout, out _); if(!sense) { mediaTest.LongBlockSize = i; break; } if(i == ushort.MaxValue) break; } DicConsole.WriteLine(); } } if(options.Debug && mediaTest.SupportsReadLong == true && mediaTest.LongBlockSize != mediaTest.BlockSize) { sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, (ushort)mediaTest.LongBlockSize, dev.Timeout, out _); if(!sense) DataFile.WriteTo("SCSI Report", "readlong10", "_debug_" + mediaTest.MediumTypeName + ".bin", "read results", buffer); } } mediaTest.MediumTypeName = mediumTypeName; mediaTest.Manufacturer = mediumManufacturer; mediaTest.Model = mediumModel; mediaTest.MediaIsRecognized = mediaIsRecognized; mediaTests.Add(mediaTest); } report.SCSI.RemovableMedias = mediaTests; } else { report.SCSI.ReadCapabilities = reporter.ReportScsi(); if(report.SCSI.ReadCapabilities.SupportsReadLong == true && report.SCSI.ReadCapabilities.LongBlockSize == report.SCSI.ReadCapabilities.BlockSize) { pressedKey = new ConsoleKeyInfo(); while(pressedKey.Key != ConsoleKey.Y && pressedKey.Key != ConsoleKey.N) { DicConsole .Write("Drive supports SCSI READ LONG but I cannot find the correct size. Do you want me to try? (This can take hours) (Y/N): "); pressedKey = System.Console.ReadKey(); DicConsole.WriteLine(); } if(pressedKey.Key == ConsoleKey.Y) { for(ushort i = (ushort)report.SCSI.ReadCapabilities.BlockSize;; i++) { DicConsole.Write("\rTrying to READ LONG with a size of {0} bytes...", i); sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, i, dev.Timeout, out _); if(!sense) { if(options.Debug) { FileStream bingo = new FileStream($"{dev.Model}_readlong.bin", FileMode.Create); bingo.Write(buffer, 0, buffer.Length); bingo.Close(); } report.SCSI.ReadCapabilities.LongBlockSize = i; break; } if(i == ushort.MaxValue) break; } DicConsole.WriteLine(); } } if(options.Debug && report.SCSI.ReadCapabilities.SupportsReadLong == true && report.SCSI.ReadCapabilities.LongBlockSize != report.SCSI.ReadCapabilities.BlockSize) { sense = dev.ReadLong10(out buffer, out senseBuffer, false, false, 0, (ushort)report.SCSI.ReadCapabilities.LongBlockSize, dev.Timeout, out _); if(!sense) DataFile.WriteTo("SCSI Report", "readlong10", "_debug_" + dev.Model + ".bin", "read results", buffer); } } break; } } break; default: throw new NotSupportedException("Unknown device type."); } FileStream jsonFs = new FileStream(jsonFile, FileMode.Create); StreamWriter jsonSw = new StreamWriter(jsonFs); jsonSw.Write(JsonConvert.SerializeObject(report, Formatting.Indented, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Ignore })); jsonSw.Close(); jsonFs.Close(); Core.Statistics.AddCommand("device-report"); using(DicContext ctx = new DicContext()) { ctx.Reports.Add(new Report(report)); ctx.SaveChanges(); } // TODO: if(Settings.Settings.Current.ShareReports) Remote.SubmitReportV2(report); } } }