mirror of
https://github.com/akuker/RASCSI.git
synced 2026-01-22 21:16:08 +00:00
Mode Sense (6) supports a 32-bit number for the maximum block count, where Mode Sense (10) supports a 64-bit number for the count. The Seagate SCSI Commands Reference Manual from 2016, has this to say about a "Short LBA mode parameter block descriptor" as returned by Mode Sense (6): > On a MODE SENSE command, if the number of logical blocks on the > medium exceeds the maximum value that is able to be specified in > the NUMBER OF LOGICAL BLOCKS field, the device server shall return > a value of FFFFFFFh. Similarly, the Read Capacity (10) command should return FFFFFFFFh if the capacity of the drive exceeds 2^32 sectors, and that a driver should then know to use Read Capacity (16) if it supports that command. There may be an unexpected side-effect if presenting a drive of more than 2^32 sectors to a legacy host that does not support devices that large. The Read Capacity commands specify the value returned as the last addressable sector on the device. This means that typically, an application which uses the value returned by that command is going to add 1 to it to get the actual number of blocks on a given device. If the program is not aware of Read Capacity (16), and is not using greater than 32-bit math, then it might report to the user that the total number of sectors on the drive as 0. I don't view this as a huge problem, however. In that case, the legacy driver wouldn't correctly support the capacity of the drive anyway, so providing that driver with a device that is 2 ^ 32 sectors or larger wouldn't make sense from the user perspective.
224 lines
5.0 KiB
C++
224 lines
5.0 KiB
C++
//---------------------------------------------------------------------------
|
||
//
|
||
// 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
|
||
// Copyright (C) 2010 Y.Sugahara
|
||
//
|
||
// Imported sava's Anex86/T98Next image and MO format support patch.
|
||
// Comments translated to english by akuker.
|
||
//
|
||
//---------------------------------------------------------------------------
|
||
|
||
#include "disk_track.h"
|
||
#include "disk_cache.h"
|
||
#include <cstdlib>
|
||
#include <cassert>
|
||
#include <algorithm>
|
||
|
||
DiskCache::DiskCache(const string& path, int size, uint64_t blocks, off_t imgoff)
|
||
: sec_path(path), sec_size(size), sec_blocks(blocks), imgoffset(imgoff)
|
||
{
|
||
assert(blocks > 0);
|
||
assert(imgoff >= 0);
|
||
}
|
||
|
||
bool DiskCache::Save()
|
||
{
|
||
// Save valid tracks
|
||
return ranges::none_of(cache.begin(), cache.end(), [this](const cache_t& c)
|
||
{ return c.disktrk != nullptr && !c.disktrk->Save(sec_path, cache_miss_write_count); });
|
||
}
|
||
|
||
shared_ptr<DiskTrack> DiskCache::GetTrack(uint64_t block)
|
||
{
|
||
// Update first
|
||
UpdateSerialNumber();
|
||
|
||
// Calculate track (fixed to 256 sectors/track)
|
||
int64_t track = block >> 8;
|
||
|
||
// Get track data
|
||
return Assign(track);
|
||
}
|
||
|
||
bool DiskCache::ReadSector(span<uint8_t> buf, uint64_t block)
|
||
{
|
||
shared_ptr<DiskTrack> disktrk = GetTrack(block);
|
||
if (disktrk == nullptr) {
|
||
return false;
|
||
}
|
||
|
||
// Read the track data to the cache
|
||
return disktrk->ReadSector(buf, block & 0xff);
|
||
}
|
||
|
||
bool DiskCache::WriteSector(span<const uint8_t> buf, uint64_t block)
|
||
{
|
||
shared_ptr<DiskTrack> disktrk = GetTrack(block);
|
||
if (disktrk == nullptr) {
|
||
return false;
|
||
}
|
||
|
||
// Write the data to the cache
|
||
return disktrk->WriteSector(buf, block & 0xff);
|
||
}
|
||
|
||
//---------------------------------------------------------------------------
|
||
//
|
||
// Track Assignment
|
||
//
|
||
//---------------------------------------------------------------------------
|
||
shared_ptr<DiskTrack> DiskCache::Assign(int64_t track)
|
||
{
|
||
assert(sec_size != 0);
|
||
assert(track >= 0);
|
||
|
||
// First, check if it is already assigned
|
||
for (cache_t& c : cache) {
|
||
if (c.disktrk && c.disktrk->GetTrack() == track) {
|
||
// Track match
|
||
c.serial = serial;
|
||
return c.disktrk;
|
||
}
|
||
}
|
||
|
||
// Next, check for empty
|
||
for (size_t i = 0; i < cache.size(); i++) {
|
||
if (cache[i].disktrk == nullptr) {
|
||
// Try loading
|
||
if (Load(static_cast<int>(i), track, nullptr)) {
|
||
// Success loading
|
||
cache[i].serial = serial;
|
||
return cache[i].disktrk;
|
||
}
|
||
|
||
// Load failed
|
||
return nullptr;
|
||
}
|
||
}
|
||
|
||
// Finally, find the youngest serial number and delete it
|
||
|
||
// Set index 0 as candidate c
|
||
uint32_t s = cache[0].serial;
|
||
size_t c = 0;
|
||
|
||
// Compare candidate with serial and update to smaller one
|
||
for (size_t i = 0; i < cache.size(); i++) {
|
||
assert(cache[i].disktrk);
|
||
|
||
// Compare and update the existing serial
|
||
if (cache[i].serial < s) {
|
||
s = cache[i].serial;
|
||
c = i;
|
||
}
|
||
}
|
||
|
||
// Save this track
|
||
if (!cache[c].disktrk->Save(sec_path, cache_miss_write_count)) {
|
||
return nullptr;
|
||
}
|
||
|
||
// Delete this track
|
||
shared_ptr<DiskTrack> disktrk = cache[c].disktrk;
|
||
cache[c].disktrk.reset();
|
||
|
||
if (Load(static_cast<int>(c), track, disktrk)) {
|
||
// Successful loading
|
||
cache[c].serial = serial;
|
||
return cache[c].disktrk;
|
||
}
|
||
|
||
// Load failed
|
||
return nullptr;
|
||
}
|
||
|
||
//---------------------------------------------------------------------------
|
||
//
|
||
// Load cache
|
||
//
|
||
//---------------------------------------------------------------------------
|
||
bool DiskCache::Load(int index, int64_t track, shared_ptr<DiskTrack> disktrk)
|
||
{
|
||
assert(index >= 0 && index < static_cast<int>(cache.size()));
|
||
assert(track >= 0);
|
||
assert(cache[index].disktrk == nullptr);
|
||
|
||
// Get the number of sectors on this track
|
||
int64_t sectors = sec_blocks - (track << 8);
|
||
assert(sectors > 0);
|
||
if (sectors > 0x100) {
|
||
sectors = 0x100;
|
||
}
|
||
|
||
if (disktrk == nullptr) {
|
||
disktrk = make_shared<DiskTrack>();
|
||
}
|
||
|
||
disktrk->Init(track, sec_size, sectors, cd_raw, imgoffset);
|
||
|
||
// Try loading
|
||
if (!disktrk->Load(sec_path, cache_miss_read_count)) {
|
||
++read_error_count;
|
||
|
||
return false;
|
||
}
|
||
|
||
// Allocation successful, work set
|
||
cache[index].disktrk = disktrk;
|
||
|
||
return true;
|
||
}
|
||
|
||
void DiskCache::UpdateSerialNumber()
|
||
{
|
||
// Update and do nothing except 0
|
||
serial++;
|
||
if (serial != 0) {
|
||
return;
|
||
}
|
||
|
||
// Clear serial of all caches
|
||
for (cache_t& c : cache) {
|
||
c.serial = 0;
|
||
}
|
||
}
|
||
|
||
vector<PbStatistics> DiskCache::GetStatistics(bool is_read_only) const
|
||
{
|
||
vector<PbStatistics> statistics;
|
||
|
||
PbStatistics s;
|
||
|
||
s.set_category(PbStatisticsCategory::CATEGORY_INFO);
|
||
|
||
s.set_key(CACHE_MISS_READ_COUNT);
|
||
s.set_value(cache_miss_read_count);
|
||
statistics.push_back(s);
|
||
|
||
if (!is_read_only) {
|
||
s.set_key(CACHE_MISS_WRITE_COUNT);
|
||
s.set_value(cache_miss_write_count);
|
||
statistics.push_back(s);
|
||
}
|
||
|
||
s.set_category(PbStatisticsCategory::CATEGORY_ERROR);
|
||
|
||
s.set_key(READ_ERROR_COUNT);
|
||
s.set_value(read_error_count);
|
||
statistics.push_back(s);
|
||
|
||
if (!is_read_only) {
|
||
s.set_key(WRITE_ERROR_COUNT);
|
||
s.set_value(write_error_count);
|
||
statistics.push_back(s);
|
||
}
|
||
|
||
return statistics;
|
||
}
|