// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : SyQuest.cs // Author(s) : Natalia Portillo // // Component : SyQuest vendor commands. // // --[ Description ] ---------------------------------------------------------- // // Contains vendor commands for SyQuest SCSI devices. // // --[ License ] -------------------------------------------------------------- // // 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 // License, or (at your option) any later version. // // 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. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, see . // // ---------------------------------------------------------------------------- // Copyright © 2011-2018 Natalia Portillo // ****************************************************************************/ using DiscImageChef.Console; namespace DiscImageChef.Devices { public partial class Device { /// /// Sends the SyQuest READ (6) command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ response will be stored /// Sense buffer. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. /// Starting block. /// Block size in bytes. public bool SyQuestRead6(out byte[] buffer, out byte[] senseBuffer, uint lba, uint blockSize, uint timeout, out double duration) { return SyQuestRead6(out buffer, out senseBuffer, lba, blockSize, 1, false, false, timeout, out duration); } /// /// Sends the SyQuest READ LONG (6) command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ response will be stored /// Sense buffer. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. /// Starting block. /// Block size in bytes. public bool SyQuestReadLong6(out byte[] buffer, out byte[] senseBuffer, uint lba, uint blockSize, uint timeout, out double duration) { return SyQuestRead6(out buffer, out senseBuffer, lba, blockSize, 1, false, true, timeout, out duration); } /// /// Sends the SyQuest READ (6) command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ response will be stored /// Sense buffer. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. /// Starting block. /// If set to true, block will not be transfer and will reside in the drive's buffer. /// If set to true drive will return ECC bytes and disable error detection. /// Block size in bytes. /// How many blocks to read. public bool SyQuestRead6(out byte[] buffer, out byte[] senseBuffer, uint lba, uint blockSize, byte transferLength, bool inhibitDma, bool readLong, uint timeout, out double duration) { senseBuffer = new byte[32]; byte[] cdb = new byte[6]; bool sense; cdb[0] = (byte)ScsiCommands.Read6; cdb[1] = (byte)((lba & 0x1F0000) >> 16); cdb[2] = (byte)((lba & 0xFF00) >> 8); cdb[3] = (byte)(lba & 0xFF); cdb[4] = transferLength; if(inhibitDma) cdb[5] += 0x80; if(readLong) cdb[5] += 0x40; if(!inhibitDma && !readLong) if(transferLength == 0) buffer = new byte[256 * blockSize]; else buffer = new byte[transferLength * blockSize]; else if(readLong) { buffer = new byte[blockSize]; cdb[4] = 1; } else buffer = new byte[0]; if(!inhibitDma) lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense); else lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.None, out duration, out sense); error = lastError != 0; DicConsole.DebugWriteLine("SCSI Device", "SYQUEST READ (6) took {0} ms.", duration); return sense; } /// /// Requests the usage, seek and error counters, and resets them /// /// Buffer. /// Sense buffer. /// Timeout. /// Duration. public bool SyQuestReadUsageCounter(out byte[] buffer, out byte[] senseBuffer, uint timeout, out double duration) { return AdaptecReadUsageCounter(out buffer, out senseBuffer, false, timeout, out duration); } /// /// Sends the SyQuest READ LONG (10) command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ response will be stored /// Sense buffer. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. /// Starting block. /// Block size in bytes. public bool SyQuestReadLong10(out byte[] buffer, out byte[] senseBuffer, uint lba, uint blockSize, uint timeout, out double duration) { return SyQuestRead10(out buffer, out senseBuffer, lba, blockSize, 1, false, true, timeout, out duration); } /// /// Sends the SyQuest READ (10) command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ response will be stored /// Sense buffer. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. /// Starting block. /// If set to true, block will not be transfer and will reside in the drive's buffer. /// If set to true drive will return ECC bytes and disable error detection. /// Block size in bytes. /// How many blocks to read. public bool SyQuestRead10(out byte[] buffer, out byte[] senseBuffer, uint lba, uint blockSize, ushort transferLength, bool inhibitDma, bool readLong, uint timeout, out double duration) { senseBuffer = new byte[32]; byte[] cdb = new byte[10]; bool sense; cdb[0] = (byte)ScsiCommands.Read10; cdb[2] = (byte)((lba & 0xFF000000) >> 24); cdb[3] = (byte)((lba & 0xFF0000) >> 16); cdb[4] = (byte)((lba & 0xFF00) >> 8); cdb[5] = (byte)(lba & 0xFF); cdb[7] = (byte)((transferLength & 0xFF00) >> 8); cdb[8] = (byte)(transferLength & 0xFF); if(inhibitDma) cdb[9] += 0x80; if(readLong) cdb[9] += 0x40; if(!inhibitDma && !readLong) buffer = new byte[transferLength * blockSize]; else if(readLong) { buffer = new byte[blockSize]; cdb[4] = 1; } else buffer = new byte[0]; if(!inhibitDma) lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense); else lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.None, out duration, out sense); error = lastError != 0; DicConsole.DebugWriteLine("SCSI Device", "SYQUEST READ (10) took {0} ms.", duration); return sense; } } }