//--------------------------------------------------------------------------- // // SCSI Target Emulator RaSCSI (*^..^*) // for Raspberry Pi // // Copyright (C) 2001-2006 PI.(ytanaka@ipc-tokai.or.jp) // Copyright (C) 2014-2020 GIMONS // Copyright (C) akuker // // Licensed under the BSD 3-Clause License. // See LICENSE file in the project root folder. // // [ SCSI hard disk ] // //--------------------------------------------------------------------------- #include "scsihd.h" #include "fileio.h" #include "exceptions.h" #include #include "../rascsi.h" #define DEFAULT_PRODUCT "SCSI HD" //=========================================================================== // // SCSI Hard Disk // //=========================================================================== //--------------------------------------------------------------------------- // // Constructor // //--------------------------------------------------------------------------- SCSIHD::SCSIHD(bool removable) : Disk(removable ? "SCRM" : "SCHD") { } void SCSIHD::FinalizeSetup(const Filepath &path, off_t size) { // 2TB is the current maximum if (size > 2LL * 1024 * 1024 * 1024 * 1024) { throw io_exception("File size must not exceed 2 TB"); } // For non-removable media drives set the default product name based on the drive capacity if (!IsRemovable()) { uint64_t capacity = GetBlockCount() * GetSectorSizeInBytes(); string unit; if (capacity >= 1000000) { capacity /= 1000000; unit = "MB"; } else { capacity /= 1000; unit = "KB"; } stringstream product; product << DEFAULT_PRODUCT << " " << capacity << " " << unit; SetProduct(product.str(), false); } SetReadOnly(false); SetProtectable(true); SetProtected(false); Disk::Open(path); FileSupport::SetPath(path); } //--------------------------------------------------------------------------- // // Reset // //--------------------------------------------------------------------------- void SCSIHD::Reset() { // Unlock and release attention SetLocked(false); SetAttn(false); // No reset, clear code SetReset(false); SetStatusCode(STATUS_NOERROR); } //--------------------------------------------------------------------------- // // Open // //--------------------------------------------------------------------------- void SCSIHD::Open(const Filepath& path) { ASSERT(!IsReady()); // Open as read-only Fileio fio; if (!fio.Open(path, Fileio::ReadOnly)) { throw file_not_found_exception("Can't open SCSI hard disk file"); } // Get file size off_t size = fio.GetFileSize(); fio.Close(); // Sector size (default 512 bytes) and number of blocks SetSectorSizeInBytes(GetConfiguredSectorSize() ? GetConfiguredSectorSize() : 512, false); SetBlockCount((DWORD)(size >> GetSectorSizeShiftCount())); // Effective size must be a multiple of the sector size size = (size / GetSectorSizeInBytes()) * GetSectorSizeInBytes(); FinalizeSetup(path, size); } //--------------------------------------------------------------------------- // // INQUIRY // //--------------------------------------------------------------------------- int SCSIHD::Inquiry(const DWORD *cdb, BYTE *buf) { ASSERT(cdb); ASSERT(buf); // EVPD check if (cdb[1] & 0x01) { SetStatusCode(STATUS_INVALIDCDB); return 0; } // Ready check (Error if no image file) if (!IsReady()) { SetStatusCode(STATUS_NOTREADY); return 0; } // Basic data // buf[0] ... Direct Access Device // buf[1] ... Bit 7 set means removable // buf[2] ... SCSI-2 compliant command system // buf[3] ... SCSI-2 compliant Inquiry response // buf[4] ... Inquiry additional data memset(buf, 0, 8); buf[1] = IsRemovable() ? 0x80 : 0x00; buf[2] = 0x02; buf[3] = 0x02; buf[4] = 28 + 3; // Value close to real HDD // Padded vendor, product, revision memcpy(&buf[8], GetPaddedName().c_str(), 28); // Size of data that can be returned int size = (buf[4] + 5); // Limit if the other buffer is small if (size > (int)cdb[4]) { size = (int)cdb[4]; } return size; } //--------------------------------------------------------------------------- // // MODE SELECT // //--------------------------------------------------------------------------- bool SCSIHD::ModeSelect(const DWORD *cdb, const BYTE *buf, int length) { int size; ASSERT(buf); ASSERT(length >= 0); // PF if (cdb[1] & 0x10) { // Mode Parameter header if (length >= 12) { // Check the block length bytes size = 1 << GetSectorSizeShiftCount(); if (buf[9] != (BYTE)(size >> 16) || buf[10] != (BYTE)(size >> 8) || buf[11] != (BYTE)size) { // currently does not allow changing sector length SetStatusCode(STATUS_INVALIDPRM); return false; } buf += 12; length -= 12; } // Parsing the page while (length > 0) { // Get page BYTE page = buf[0]; switch (page) { // format device case 0x03: // check the number of bytes in the physical sector size = 1 << GetSectorSizeShiftCount(); if (buf[0xc] != (BYTE)(size >> 8) || buf[0xd] != (BYTE)size) { // currently does not allow changing sector length SetStatusCode(STATUS_INVALIDPRM); return false; } break; // CD-ROM Parameters // TODO Move to scsicd.cpp // According to the SONY CDU-541 manual, Page code 8 is supposed // to set the Logical Block Adress Format, as well as the // inactivity timer multiplier case 0x08: // Debug code for Issue #2: // https://github.com/akuker/RASCSI/issues/2 LOGWARN("[Unhandled page code] Received mode page code 8 with total length %d\n ", length); for (int i = 0; i