//--------------------------------------------------------------------------- // // X68000 EMULATOR "XM6" // // Copyright (C) 2001-2006 PI.(ytanaka@ipc-tokai.or.jp) // Copyright (C) 2014-2020 GIMONS // // XM6i // Copyright (C) 2010-2015 isaki@NetBSD.org // // Imported sava's Anex86/T98Next image and MO format support patch. // Comments translated to english by akuker. // //--------------------------------------------------------------------------- #pragma once #include "scsi.h" #include "device_factory.h" #include "disk_track.h" #include "disk_cache.h" #include "interfaces/scsi_block_commands.h" #include "storage_device.h" #include #include #include using namespace std; class Disk : public StorageDevice, private ScsiBlockCommands { enum access_mode { RW6, RW10, RW16, SEEK6, SEEK10 }; Dispatcher dispatcher; unique_ptr cache; // The supported configurable sector sizes, empty if not configurable unordered_set sector_sizes; uint32_t configured_sector_size = 0; // Sector size shift count (9=512, 10=1024, 11=2048, 12=4096) uint32_t size_shift_count = 0; public: Disk(PbDeviceType, int); ~Disk() override; bool Dispatch(scsi_command) override; bool Eject(bool) override; // Command helpers virtual int WriteCheck(uint64_t); virtual void Write(const vector&, const vector&, uint64_t); virtual int Read(const vector&, vector& , uint64_t); uint32_t GetSectorSizeInBytes() const; bool IsSectorSizeConfigurable() const { return !sector_sizes.empty(); } bool SetConfiguredSectorSize(const DeviceFactory&, uint32_t); void FlushCache() override; private: using super = StorageDevice; // Commands covered by the SCSI specifications (see https://www.t10.org/drafts.htm) void StartStopUnit(); void PreventAllowMediumRemoval(); void SynchronizeCache(); void ReadDefectData10(); virtual void Read6() { Read(RW6); } void Read10() override { Read(RW10); } void Read16() override { Read(RW16); } virtual void Write6() { Write(RW6); } void Write10() override { Write(RW10); } void Write16() override { Write(RW16); } void Verify10() { Verify(RW10); } void Verify16() { Verify(RW16); } void Seek(); void Seek10(); void ReadCapacity10() override; void ReadCapacity16() override; void FormatUnit() override; void Seek6(); void Read(access_mode); void Write(access_mode); void Verify(access_mode); void ReadWriteLong10(); void ReadWriteLong16(); void ReadCapacity16_ReadLong16(); void ValidateBlockAddress(access_mode) const; bool CheckAndGetStartAndCount(uint64_t&, uint32_t&, access_mode) const; int ModeSense6(const vector&, vector&) const override; int ModeSense10(const vector&, vector&) const override; static const unordered_map shift_counts; protected: void SetUpCache(off_t, bool = false); void ResizeCache(const string&, bool); void SetUpModePages(map>&, int, bool) const override; void AddErrorPage(map>&, bool) const; virtual void AddFormatPage(map>&, bool) const; virtual void AddDrivePage(map>&, bool) const; void AddCachePage(map>&, bool) const; unordered_set GetSectorSizes() const; void SetSectorSizes(const unordered_set& sizes) { sector_sizes = sizes; } void SetSectorSizeInBytes(uint32_t); uint32_t GetSectorSizeShiftCount() const { return size_shift_count; } void SetSectorSizeShiftCount(uint32_t count) { size_shift_count = count; } uint32_t GetConfiguredSectorSize() const; static uint32_t CalculateShiftCount(uint32_t); };