// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : HP.cs // Author(s) : Natalia Portillo // // Component : Hewlett-Packard vendor commands. // // --[ Description ] ---------------------------------------------------------- // // Contains vendor commands for Hewlett-Packard 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-2016 Natalia Portillo // ****************************************************************************/ using DiscImageChef.Console; namespace DiscImageChef.Devices { public partial class Device { /// /// Sends the HP READ LONG vendor command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ LONG response will be stored /// Sense buffer. /// If set to true address contain two's complement offset from last read address. /// PBA/LBA to read. /// How many bytes per block. /// If set to true address contain physical block address. /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. public bool HPReadLong(out byte[] buffer, out byte[] senseBuffer, bool relAddr, uint address, ushort blockBytes, bool pba, uint timeout, out double duration) { return HPReadLong(out buffer, out senseBuffer, relAddr, address, 0, blockBytes, pba, false, timeout, out duration); } /// /// Sends the HP READ LONG vendor command /// /// true if the command failed and contains the sense buffer. /// Buffer where the SCSI READ LONG response will be stored /// Sense buffer. /// If set to true address contain two's complement offset from last read address. /// PBA/LBA to read. /// How many blocks/bytes to read. /// How many bytes per block. /// If set to true address contain physical block address. /// If set to true is a count of secors to read. Otherwise it will be ignored /// Timeout in seconds. /// Duration in milliseconds it took for the device to execute the command. public bool HPReadLong(out byte[] buffer, out byte[] senseBuffer, bool relAddr, uint address, ushort transferLen, ushort blockBytes, bool pba, bool sectorCount, uint timeout, out double duration) { senseBuffer = new byte[32]; byte[] cdb = new byte[10]; bool sense; cdb[0] = (byte)ScsiCommands.ReadLong; if(relAddr) cdb[1] += 0x01; cdb[2] = (byte)((address & 0xFF000000) >> 24); cdb[3] = (byte)((address & 0xFF0000) >> 16); cdb[4] = (byte)((address & 0xFF00) >> 8); cdb[5] = (byte)(address & 0xFF); cdb[7] = (byte)((transferLen & 0xFF00) >> 8); cdb[8] = (byte)(transferLen & 0xFF); if(pba) cdb[9] += 0x80; if(sectorCount) cdb[9] += 0x40; if(sectorCount) buffer = new byte[blockBytes * transferLen]; else buffer = new byte[transferLen]; lastError = SendScsiCommand(cdb, ref buffer, out senseBuffer, timeout, ScsiDirection.In, out duration, out sense); error = lastError != 0; DicConsole.DebugWriteLine("SCSI Device", "HP READ LONG took {0} ms.", duration); return sense; } } }