// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : Ata48.cs // Author(s) : Natalia Portillo // // Component : DiscImageChef device testing. // // --[ 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 DiscImageChef.Console; using DiscImageChef.Decoders.ATA; using DiscImageChef.Devices; namespace DiscImageChef.Tests.Devices.ATA { static class Ata48 { internal static void Menu(string devPath, Device dev) { while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Send a 48-bit ATA command to the device:"); DicConsole.WriteLine("1.- Send GET NATIVE MAX ADDRESS EXT command."); DicConsole.WriteLine("2.- Send READ DMA EXT command."); DicConsole.WriteLine("3.- Send READ LOG DMA EXT command."); DicConsole.WriteLine("4.- Send READ LOG EXT command."); DicConsole.WriteLine("5.- Send READ MATIVE MAX ADDRESS EXT command."); DicConsole.WriteLine("6.- Send READ MULTIPLE EXT command."); DicConsole.WriteLine("7.- Send READ SECTORS EXT command."); DicConsole.WriteLine("0.- Return to ATA commands menu."); DicConsole.Write("Choose: "); string strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out int item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to ATA commands menu..."); return; case 1: GetNativeMaxAddressExt(devPath, dev); continue; case 2: ReadDmaExt(devPath, dev); continue; case 3: ReadLogDmaExt(devPath, dev); continue; case 4: ReadLogExt(devPath, dev); continue; case 5: ReadNativeMaxAddressExt(devPath, dev); continue; case 6: ReadMultipleExt(devPath, dev); continue; case 7: ReadSectorsExt(devPath, dev); continue; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); continue; } } } static void GetNativeMaxAddressExt(string devPath, Device dev) { start: System.Console.Clear(); bool sense = dev.GetNativeMaxAddressExt(out ulong lba, out AtaErrorRegistersLba48 errorRegisters, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending GET NATIVE MAX ADDRESS EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Max LBA is {0}.", lba); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Decode error registers."); DicConsole.WriteLine("2.- Send command again."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); string strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out int item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("GET NATIVE MAX ADDRESS EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: goto start; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadDmaExt(string devPath, Device dev) { ulong lba = 0; ushort count = 1; string strDev; int item; parameters: while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Parameters for READ DMA EXT command:"); DicConsole.WriteLine("LBA: {0}", lba); DicConsole.WriteLine("Count: {0}", count); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Change parameters."); DicConsole.WriteLine("2.- Send command with these parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: DicConsole.Write("What logical block address?: "); strDev = System.Console.ReadLine(); if(!ulong.TryParse(strDev, out lba)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); lba = 0; System.Console.ReadKey(); continue; } if(lba > 0xFFFFFFFFFFFF) { DicConsole .WriteLine("Logical block address cannot be bigger than {0}. Setting it to {0}...", 0xFFFFFFFFFFFF); lba = 0xFFFFFFFFFFFF; } DicConsole.Write("How many sectors?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out count)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); count = 0; System.Console.ReadKey(); } break; case 2: goto start; } } start: System.Console.Clear(); bool sense = dev.ReadDma(out byte[] buffer, out AtaErrorRegistersLba48 errorRegisters, lba, count, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ DMA EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Buffer is {0} bytes.", buffer?.Length.ToString() ?? "null"); DicConsole.WriteLine("Buffer is null or empty? {0}", ArrayHelpers.ArrayIsNullOrEmpty(buffer)); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Print buffer."); DicConsole.WriteLine("2.- Decode error registers."); DicConsole.WriteLine("3.- Send command again."); DicConsole.WriteLine("4.- Change parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ DMA EXT response:"); if(buffer != null) PrintHex.PrintHexArray(buffer, 64); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ DMA EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 3: goto start; case 4: goto parameters; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadLogExt(string devPath, Device dev) { byte address = 0; ushort page = 0; ushort count = 1; string strDev; int item; parameters: while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Parameters for READ LOG EXT command:"); DicConsole.WriteLine("Log address: {0}", address); DicConsole.WriteLine("Page number: {0}", page); DicConsole.WriteLine("Count: {0}", count); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Change parameters."); DicConsole.WriteLine("2.- Send command with these parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: DicConsole.Write("What log address?: "); strDev = System.Console.ReadLine(); if(!byte.TryParse(strDev, out address)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); address = 0; System.Console.ReadKey(); continue; } DicConsole.Write("What page number?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out page)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); page = 0; System.Console.ReadKey(); continue; } DicConsole.Write("How many sectors?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out count)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); count = 0; System.Console.ReadKey(); } break; case 2: goto start; } } start: System.Console.Clear(); bool sense = dev.ReadLog(out byte[] buffer, out AtaErrorRegistersLba48 errorRegisters, address, page, count, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ LOG EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Buffer is {0} bytes.", buffer?.Length.ToString() ?? "null"); DicConsole.WriteLine("Buffer is null or empty? {0}", ArrayHelpers.ArrayIsNullOrEmpty(buffer)); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Print buffer."); DicConsole.WriteLine("2.- Decode error registers."); DicConsole.WriteLine("3.- Send command again."); DicConsole.WriteLine("4.- Change parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ LOG EXT response:"); if(buffer != null) PrintHex.PrintHexArray(buffer, 64); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ LOG EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 3: goto start; case 4: goto parameters; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadLogDmaExt(string devPath, Device dev) { byte address = 0; ushort page = 0; ushort count = 1; string strDev; int item; parameters: while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Parameters for READ LOG DMA EXT command:"); DicConsole.WriteLine("Log address: {0}", address); DicConsole.WriteLine("Page number: {0}", page); DicConsole.WriteLine("Count: {0}", count); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Change parameters."); DicConsole.WriteLine("2.- Send command with these parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: DicConsole.Write("What log address?: "); strDev = System.Console.ReadLine(); if(!byte.TryParse(strDev, out address)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); address = 0; System.Console.ReadKey(); continue; } DicConsole.Write("What page number?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out page)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); page = 0; System.Console.ReadKey(); continue; } DicConsole.Write("How many sectors?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out count)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); count = 0; System.Console.ReadKey(); } break; case 2: goto start; } } start: System.Console.Clear(); bool sense = dev.ReadLogDma(out byte[] buffer, out AtaErrorRegistersLba48 errorRegisters, address, page, count, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ LOG DMA EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Buffer is {0} bytes.", buffer?.Length.ToString() ?? "null"); DicConsole.WriteLine("Buffer is null or empty? {0}", ArrayHelpers.ArrayIsNullOrEmpty(buffer)); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Print buffer."); DicConsole.WriteLine("2.- Decode error registers."); DicConsole.WriteLine("3.- Send command again."); DicConsole.WriteLine("4.- Change parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ LOG DMA EXT response:"); if(buffer != null) PrintHex.PrintHexArray(buffer, 64); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ LOG DMA EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 3: goto start; case 4: goto parameters; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadMultipleExt(string devPath, Device dev) { ulong lba = 0; ushort count = 1; string strDev; int item; parameters: while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Parameters for READ MULTIPLE EXT command:"); DicConsole.WriteLine("LBA: {0}", lba); DicConsole.WriteLine("Count: {0}", count); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Change parameters."); DicConsole.WriteLine("2.- Send command with these parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: DicConsole.Write("What logical block address?: "); strDev = System.Console.ReadLine(); if(!ulong.TryParse(strDev, out lba)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); lba = 0; System.Console.ReadKey(); continue; } if(lba > 0xFFFFFFFFFFFF) { DicConsole .WriteLine("Logical block address cannot be bigger than {0}. Setting it to {0}...", 0xFFFFFFFFFFFF); lba = 0xFFFFFFFFFFFF; } DicConsole.Write("How many sectors?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out count)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); count = 0; System.Console.ReadKey(); } break; case 2: goto start; } } start: System.Console.Clear(); bool sense = dev.ReadMultiple(out byte[] buffer, out AtaErrorRegistersLba48 errorRegisters, lba, count, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ MULTIPLE EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Buffer is {0} bytes.", buffer?.Length.ToString() ?? "null"); DicConsole.WriteLine("Buffer is null or empty? {0}", ArrayHelpers.ArrayIsNullOrEmpty(buffer)); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Print buffer."); DicConsole.WriteLine("2.- Decode error registers."); DicConsole.WriteLine("3.- Send command again."); DicConsole.WriteLine("4.- Change parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ MULTIPLE EXT response:"); if(buffer != null) PrintHex.PrintHexArray(buffer, 64); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ MULTIPLE EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 3: goto start; case 4: goto parameters; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadNativeMaxAddressExt(string devPath, Device dev) { start: System.Console.Clear(); bool sense = dev.ReadNativeMaxAddress(out ulong lba, out AtaErrorRegistersLba48 errorRegisters, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ NATIVE MAX ADDRESS EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Max LBA is {0}.", lba); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Decode error registers."); DicConsole.WriteLine("2.- Send command again."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); string strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out int item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ NATIVE MAX ADDRESS status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: goto start; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } static void ReadSectorsExt(string devPath, Device dev) { ulong lba = 0; ushort count = 1; string strDev; int item; parameters: while(true) { System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Parameters for READ SECTORS EXT command:"); DicConsole.WriteLine("LBA: {0}", lba); DicConsole.WriteLine("Count: {0}", count); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Change parameters."); DicConsole.WriteLine("2.- Send command with these parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); continue; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: DicConsole.Write("What logical block address?: "); strDev = System.Console.ReadLine(); if(!ulong.TryParse(strDev, out lba)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); lba = 0; System.Console.ReadKey(); continue; } if(lba > 0xFFFFFFFFFFFF) { DicConsole .WriteLine("Logical block address cannot be bigger than {0}. Setting it to {0}...", 0xFFFFFFFFFFFF); lba = 0xFFFFFFFFFFFF; } DicConsole.Write("How many sectors?: "); strDev = System.Console.ReadLine(); if(!ushort.TryParse(strDev, out count)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); count = 0; System.Console.ReadKey(); } break; case 2: goto start; } } start: System.Console.Clear(); bool sense = dev.Read(out byte[] buffer, out AtaErrorRegistersLba48 errorRegisters, lba, count, dev.Timeout, out double duration); menu: DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("Sending READ SECTORS EXT to the device:"); DicConsole.WriteLine("Command took {0} ms.", duration); DicConsole.WriteLine("Sense is {0}.", sense); DicConsole.WriteLine("Buffer is {0} bytes.", buffer?.Length.ToString() ?? "null"); DicConsole.WriteLine("Buffer is null or empty? {0}", ArrayHelpers.ArrayIsNullOrEmpty(buffer)); DicConsole.WriteLine(); DicConsole.WriteLine("Choose what to do:"); DicConsole.WriteLine("1.- Print buffer."); DicConsole.WriteLine("2.- Decode error registers."); DicConsole.WriteLine("3.- Send command again."); DicConsole.WriteLine("4.- Change parameters."); DicConsole.WriteLine("0.- Return to 48-bit ATA commands menu."); DicConsole.Write("Choose: "); strDev = System.Console.ReadLine(); if(!int.TryParse(strDev, out item)) { DicConsole.WriteLine("Not a number. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } switch(item) { case 0: DicConsole.WriteLine("Returning to 48-bit ATA commands menu..."); return; case 1: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ SECTORS EXT response:"); if(buffer != null) PrintHex.PrintHexArray(buffer, 64); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 2: System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); DicConsole.WriteLine("READ SECTORS EXT status registers:"); DicConsole.Write("{0}", MainClass.DecodeAtaRegisters(errorRegisters)); DicConsole.WriteLine("Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); DicConsole.WriteLine("Device: {0}", devPath); goto menu; case 3: goto start; case 4: goto parameters; default: DicConsole.WriteLine("Incorrect option. Press any key to continue..."); System.Console.ReadKey(); System.Console.Clear(); goto menu; } } } }