Merge pull request #3795 from EmptyChaos/fix-diskreader
DriveReader: Fix View > Show Drives
This commit is contained in:
commit
05e1406e89
|
@ -2,8 +2,8 @@
|
||||||
// Licensed under GPLv2+
|
// Licensed under GPLv2+
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <cstring>
|
|
||||||
#include <limits>
|
#include <limits>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
@ -22,87 +22,156 @@
|
||||||
namespace DiscIO
|
namespace DiscIO
|
||||||
{
|
{
|
||||||
|
|
||||||
// Provides caching and split-operation-to-block-operations facilities.
|
|
||||||
// Used for compressed blob reading and direct drive reading.
|
|
||||||
|
|
||||||
void SectorReader::SetSectorSize(int blocksize)
|
void SectorReader::SetSectorSize(int blocksize)
|
||||||
{
|
{
|
||||||
|
m_block_size = std::max(blocksize, 0);
|
||||||
for (auto& cache_entry : m_cache)
|
for (auto& cache_entry : m_cache)
|
||||||
cache_entry.resize(blocksize);
|
{
|
||||||
|
cache_entry.Reset();
|
||||||
|
cache_entry.data.resize(m_chunk_blocks * m_block_size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
m_cache_tags.fill(std::numeric_limits<u64>::max());
|
void SectorReader::SetChunkSize(int block_cnt)
|
||||||
m_blocksize = blocksize;
|
{
|
||||||
|
m_chunk_blocks = std::max(block_cnt, 1);
|
||||||
|
// Clear cache and resize the data arrays
|
||||||
|
SetSectorSize(m_block_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
SectorReader::~SectorReader()
|
SectorReader::~SectorReader()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::vector<u8>& SectorReader::GetBlockData(u64 block_num)
|
const SectorReader::Cache* SectorReader::FindCacheLine(u64 block_num)
|
||||||
{
|
{
|
||||||
// TODO : Expand usage of the cache to more than one block :P
|
auto itr = std::find_if(m_cache.begin(), m_cache.end(), [&](const Cache& entry)
|
||||||
if (m_cache_tags[0] == block_num)
|
{
|
||||||
return m_cache[0];
|
return entry.Contains(block_num);
|
||||||
|
});
|
||||||
|
if (itr == m_cache.end())
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
GetBlock(block_num, m_cache[0].data());
|
itr->MarkUsed();
|
||||||
m_cache_tags[0] = block_num;
|
return &*itr;
|
||||||
return m_cache[0];
|
}
|
||||||
|
|
||||||
|
SectorReader::Cache* SectorReader::GetEmptyCacheLine()
|
||||||
|
{
|
||||||
|
Cache* oldest = &m_cache[0];
|
||||||
|
// Find the Least Recently Used cache line to replace.
|
||||||
|
for (auto& cache_entry : m_cache)
|
||||||
|
{
|
||||||
|
if (cache_entry.IsLessRecentlyUsedThan(*oldest))
|
||||||
|
oldest = &cache_entry;
|
||||||
|
cache_entry.ShiftLRU();
|
||||||
|
}
|
||||||
|
oldest->Reset();
|
||||||
|
return oldest;
|
||||||
|
}
|
||||||
|
|
||||||
|
const SectorReader::Cache* SectorReader::GetCacheLine(u64 block_num)
|
||||||
|
{
|
||||||
|
if (auto entry = FindCacheLine(block_num))
|
||||||
|
return entry;
|
||||||
|
|
||||||
|
// Cache miss. Fault in the missing entry.
|
||||||
|
Cache* cache = GetEmptyCacheLine();
|
||||||
|
// We only read aligned chunks, this avoids duplicate overlapping entries.
|
||||||
|
u64 chunk_idx = block_num / m_chunk_blocks;
|
||||||
|
u32 blocks_read = ReadChunk(cache->data.data(), chunk_idx);
|
||||||
|
if (!blocks_read)
|
||||||
|
return nullptr;
|
||||||
|
cache->Fill(chunk_idx * m_chunk_blocks, blocks_read);
|
||||||
|
|
||||||
|
// Secondary check for out-of-bounds read.
|
||||||
|
// If we got less than m_chunk_blocks, we may still have missed.
|
||||||
|
// We do this after the cache fill since the cache line itself is
|
||||||
|
// fine, the problem is being asked to read past the end of the disk.
|
||||||
|
return cache->Contains(block_num) ? cache : nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SectorReader::Read(u64 offset, u64 size, u8* out_ptr)
|
bool SectorReader::Read(u64 offset, u64 size, u8* out_ptr)
|
||||||
{
|
{
|
||||||
u64 startingBlock = offset / m_blocksize;
|
|
||||||
u64 remain = size;
|
u64 remain = size;
|
||||||
|
u64 block = 0;
|
||||||
int positionInBlock = (int)(offset % m_blocksize);
|
u32 position_in_block = static_cast<u32>(offset % m_block_size);
|
||||||
u64 block = startingBlock;
|
|
||||||
|
|
||||||
while (remain > 0)
|
while (remain > 0)
|
||||||
{
|
{
|
||||||
// Check if we are ready to do a large block read. > instead of >= so we don't bother if remain is only one block.
|
block = offset / m_block_size;
|
||||||
if (positionInBlock == 0 && remain > (u64)m_blocksize)
|
|
||||||
{
|
|
||||||
u64 num_blocks = remain / m_blocksize;
|
|
||||||
ReadMultipleAlignedBlocks(block, num_blocks, out_ptr);
|
|
||||||
block += num_blocks;
|
|
||||||
out_ptr += num_blocks * m_blocksize;
|
|
||||||
remain -= num_blocks * m_blocksize;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::vector<u8>& data = GetBlockData(block);
|
const Cache* cache = GetCacheLine(block);
|
||||||
|
if (!cache)
|
||||||
|
return false;
|
||||||
|
|
||||||
u32 to_copy = m_blocksize - positionInBlock;
|
// Cache entries are aligned chunks, we may not want to read from the start
|
||||||
if (to_copy >= remain)
|
u32 read_offset = static_cast<u32>(block - cache->block_idx) * m_block_size + position_in_block;
|
||||||
{
|
u32 can_read = m_block_size * cache->num_blocks - read_offset;
|
||||||
// Yay, we are done!
|
u32 was_read = static_cast<u32>(std::min<u64>(can_read, remain));
|
||||||
std::copy(data.begin() + positionInBlock, data.begin() + positionInBlock + remain, out_ptr);
|
|
||||||
return true;
|
std::copy(cache->data.begin() + read_offset,
|
||||||
}
|
cache->data.begin() + read_offset + was_read,
|
||||||
else
|
out_ptr);
|
||||||
{
|
|
||||||
std::copy(data.begin() + positionInBlock, data.begin() + positionInBlock + to_copy, out_ptr);
|
offset += was_read;
|
||||||
out_ptr += to_copy;
|
out_ptr += was_read;
|
||||||
remain -= to_copy;
|
remain -= was_read;
|
||||||
positionInBlock = 0;
|
position_in_block = 0;
|
||||||
block++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SectorReader::ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr)
|
// Crap default implementation if not overridden.
|
||||||
|
bool SectorReader::ReadMultipleAlignedBlocks(u64 block_num, u64 cnt_blocks, u8* out_ptr)
|
||||||
{
|
{
|
||||||
for (u64 i = 0; i < num_blocks; i++)
|
for (u64 i = 0; i < cnt_blocks; ++i)
|
||||||
{
|
{
|
||||||
const std::vector<u8>& data = GetBlockData(block_num + i);
|
if (!GetBlock(block_num + i, out_ptr))
|
||||||
const u64 offset = i * m_blocksize;
|
return false;
|
||||||
|
out_ptr += m_block_size;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
std::copy(data.begin(), data.end(), out_ptr + offset);
|
u32 SectorReader::ReadChunk(u8* buffer, u64 chunk_num)
|
||||||
|
{
|
||||||
|
u64 block_num = chunk_num * m_chunk_blocks;
|
||||||
|
u32 cnt_blocks = m_chunk_blocks;
|
||||||
|
|
||||||
|
// If we are reading the end of a disk, there may not be enough blocks to
|
||||||
|
// read a whole chunk. We need to clamp down in that case.
|
||||||
|
u64 end_block = GetDataSize() / m_block_size;
|
||||||
|
if (end_block)
|
||||||
|
cnt_blocks = static_cast<u32>(std::min<u64>(m_chunk_blocks, end_block - block_num));
|
||||||
|
|
||||||
|
if (ReadMultipleAlignedBlocks(block_num, cnt_blocks, buffer))
|
||||||
|
{
|
||||||
|
if (cnt_blocks < m_chunk_blocks)
|
||||||
|
{
|
||||||
|
std::fill(buffer + cnt_blocks * m_block_size,
|
||||||
|
buffer + m_chunk_blocks * m_block_size,
|
||||||
|
0u);
|
||||||
|
}
|
||||||
|
return cnt_blocks;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
// end_block may be zero on real disks if we fail to get the media size.
|
||||||
|
// We have to fallback to probing the disk instead.
|
||||||
|
if (!end_block)
|
||||||
|
{
|
||||||
|
for (u32 i = 0; i < cnt_blocks; ++i)
|
||||||
|
{
|
||||||
|
if (!GetBlock(block_num + i, buffer))
|
||||||
|
{
|
||||||
|
std::fill(buffer, buffer + (cnt_blocks - i) * m_block_size, 0u);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
buffer += m_block_size;
|
||||||
|
}
|
||||||
|
return cnt_blocks;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<IBlobReader> CreateBlobReader(const std::string& filename)
|
std::unique_ptr<IBlobReader> CreateBlobReader(const std::string& filename)
|
||||||
|
|
|
@ -50,32 +50,112 @@ protected:
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
// Provides caching and split-operation-to-block-operations facilities.
|
// Provides caching and byte-operation-to-block-operations facilities.
|
||||||
// Used for compressed blob reading and direct drive reading.
|
// Used for compressed blob and direct drive reading.
|
||||||
// Currently only uses a single entry cache.
|
// NOTE: GetDataSize() is expected to be evenly divisible by the sector size.
|
||||||
// Multi-block reads are not cached.
|
|
||||||
class SectorReader : public IBlobReader
|
class SectorReader : public IBlobReader
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
virtual ~SectorReader();
|
virtual ~SectorReader() = 0;
|
||||||
|
|
||||||
bool Read(u64 offset, u64 size, u8 *out_ptr) override;
|
bool Read(u64 offset, u64 size, u8* out_ptr) override;
|
||||||
friend class DriveReader;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void SetSectorSize(int blocksize);
|
void SetSectorSize(int blocksize);
|
||||||
virtual void GetBlock(u64 block_num, u8 *out) = 0;
|
int GetSectorSize() const
|
||||||
// This one is uncached. The default implementation is to simply call GetBlockData multiple times and memcpy.
|
{
|
||||||
virtual bool ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8 *out_ptr);
|
return m_block_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the chunk size -> the number of blocks to read at a time.
|
||||||
|
// Default value is 1 but that is too low for physical devices
|
||||||
|
// like CDROMs. Setting this to a higher value helps reduce seeking
|
||||||
|
// and IO overhead by batching reads. Do not set it too high either
|
||||||
|
// as large reads are slow and will take too long to resolve.
|
||||||
|
void SetChunkSize(int blocks);
|
||||||
|
int GetChunkSize() const
|
||||||
|
{
|
||||||
|
return m_chunk_blocks;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read a single block/sector.
|
||||||
|
virtual bool GetBlock(u64 block_num, u8* out) = 0;
|
||||||
|
|
||||||
|
// Read multiple contiguous blocks.
|
||||||
|
// Default implementation just calls GetBlock in a loop, it should be
|
||||||
|
// overridden in derived classes where possible.
|
||||||
|
virtual bool ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// A reference returned by GetBlockData is invalidated as soon as GetBlockData, Read, or ReadMultipleAlignedBlocks is called again.
|
struct Cache
|
||||||
const std::vector<u8>& GetBlockData(u64 block_num);
|
{
|
||||||
|
std::vector<u8> data;
|
||||||
|
u64 block_idx = 0;
|
||||||
|
u32 num_blocks = 0;
|
||||||
|
|
||||||
enum { CACHE_SIZE = 32 };
|
// [Pseudo-] Least Recently Used Shift Register
|
||||||
int m_blocksize;
|
// When an empty cache line is needed, the line with the lowest value
|
||||||
std::array<std::vector<u8>, CACHE_SIZE> m_cache;
|
// is taken and reset; the LRU register is then shifted down 1 place
|
||||||
std::array<u64, CACHE_SIZE> m_cache_tags;
|
// on all lines (low bit discarded). When a line is used, the high bit
|
||||||
|
// is set marking it as most recently used.
|
||||||
|
u32 lru_sreg = 0;
|
||||||
|
|
||||||
|
void Reset()
|
||||||
|
{
|
||||||
|
block_idx = 0;
|
||||||
|
num_blocks = 0;
|
||||||
|
lru_sreg = 0;
|
||||||
|
}
|
||||||
|
void Fill(u64 block, u32 count)
|
||||||
|
{
|
||||||
|
block_idx = block;
|
||||||
|
num_blocks = count;
|
||||||
|
// NOTE: Setting only the high bit means the newest line will
|
||||||
|
// be selected for eviction if every line in the cache was
|
||||||
|
// touched. This gives MRU behavior which is probably
|
||||||
|
// desirable in that case.
|
||||||
|
MarkUsed();
|
||||||
|
}
|
||||||
|
bool Contains(u64 block) const
|
||||||
|
{
|
||||||
|
return block >= block_idx && block - block_idx < num_blocks;
|
||||||
|
}
|
||||||
|
void MarkUsed()
|
||||||
|
{
|
||||||
|
lru_sreg |= 0x80000000;
|
||||||
|
}
|
||||||
|
void ShiftLRU()
|
||||||
|
{
|
||||||
|
lru_sreg >>= 1;
|
||||||
|
}
|
||||||
|
bool IsLessRecentlyUsedThan(const Cache& other) const
|
||||||
|
{
|
||||||
|
return lru_sreg < other.lru_sreg;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Gets the cache line that contains the given block, or nullptr.
|
||||||
|
// NOTE: The cache record only lasts until it expires (next GetEmptyCacheLine)
|
||||||
|
const Cache* FindCacheLine(u64 block_num);
|
||||||
|
|
||||||
|
// Finds the least recently used cache line, resets and returns it.
|
||||||
|
Cache* GetEmptyCacheLine();
|
||||||
|
|
||||||
|
// Combines FindCacheLine with GetEmptyCacheLine and ReadChunk.
|
||||||
|
// Always returns a valid cache line (loading the data if needed).
|
||||||
|
// May return nullptr only if the cache missed and the read failed.
|
||||||
|
const Cache* GetCacheLine(u64 block_num);
|
||||||
|
|
||||||
|
// Read all bytes from a chunk of blocks into a buffer.
|
||||||
|
// Returns the number of blocks read (may be less than m_chunk_blocks
|
||||||
|
// if chunk_num is the last chunk on the disk and the disk size is not
|
||||||
|
// evenly divisible into chunks). Returns zero if it fails.
|
||||||
|
u32 ReadChunk(u8* buffer, u64 chunk_num);
|
||||||
|
|
||||||
|
static constexpr int CACHE_LINES = 32;
|
||||||
|
u32 m_block_size = 0; // Bytes in a sector/block
|
||||||
|
u32 m_chunk_blocks = 1; // Number of sectors/blocks in a chunk
|
||||||
|
std::array<Cache, CACHE_LINES> m_cache;
|
||||||
};
|
};
|
||||||
|
|
||||||
class CBlobBigEndianReader
|
class CBlobBigEndianReader
|
||||||
|
|
|
@ -79,7 +79,7 @@ u64 CompressedBlobReader::GetBlockCompressedSize(u64 block_num) const
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CompressedBlobReader::GetBlock(u64 block_num, u8 *out_ptr)
|
bool CompressedBlobReader::GetBlock(u64 block_num, u8 *out_ptr)
|
||||||
{
|
{
|
||||||
bool uncompressed = false;
|
bool uncompressed = false;
|
||||||
u32 comp_block_size = (u32)GetBlockCompressedSize(block_num);
|
u32 comp_block_size = (u32)GetBlockCompressedSize(block_num);
|
||||||
|
@ -97,7 +97,13 @@ void CompressedBlobReader::GetBlock(u64 block_num, u8 *out_ptr)
|
||||||
memset(&m_zlib_buffer[comp_block_size], 0, m_zlib_buffer.size() - comp_block_size);
|
memset(&m_zlib_buffer[comp_block_size], 0, m_zlib_buffer.size() - comp_block_size);
|
||||||
|
|
||||||
m_file.Seek(offset, SEEK_SET);
|
m_file.Seek(offset, SEEK_SET);
|
||||||
m_file.ReadBytes(m_zlib_buffer.data(), comp_block_size);
|
if (!m_file.ReadBytes(m_zlib_buffer.data(), comp_block_size))
|
||||||
|
{
|
||||||
|
PanicAlertT("The disc image \"%s\" is truncated, some of the data is missing.",
|
||||||
|
m_file_name.c_str());
|
||||||
|
m_file.Clear();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// First, check hash.
|
// First, check hash.
|
||||||
u32 block_hash = HashAdler32(m_zlib_buffer.data(), comp_block_size);
|
u32 block_hash = HashAdler32(m_zlib_buffer.data(), comp_block_size);
|
||||||
|
@ -133,8 +139,12 @@ void CompressedBlobReader::GetBlock(u64 block_num, u8 *out_ptr)
|
||||||
}
|
}
|
||||||
inflateEnd(&z);
|
inflateEnd(&z);
|
||||||
if (uncomp_size != m_header.block_size)
|
if (uncomp_size != m_header.block_size)
|
||||||
|
{
|
||||||
PanicAlert("Wrong block size");
|
PanicAlert("Wrong block size");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u32 sub_type,
|
bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u32 sub_type,
|
||||||
|
|
|
@ -55,7 +55,7 @@ public:
|
||||||
u64 GetDataSize() const override { return m_header.data_size; }
|
u64 GetDataSize() const override { return m_header.data_size; }
|
||||||
u64 GetRawSize() const override { return m_file_size; }
|
u64 GetRawSize() const override { return m_file_size; }
|
||||||
u64 GetBlockCompressedSize(u64 block_num) const;
|
u64 GetBlockCompressedSize(u64 block_num) const;
|
||||||
void GetBlock(u64 block_num, u8* out_ptr) override;
|
bool GetBlock(u64 block_num, u8* out_ptr) override;
|
||||||
private:
|
private:
|
||||||
CompressedBlobReader(const std::string& filename);
|
CompressedBlobReader(const std::string& filename);
|
||||||
|
|
||||||
|
|
|
@ -18,6 +18,16 @@
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
|
#else
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <stdio.h> // fileno
|
||||||
|
#if defined __linux__
|
||||||
|
#include <linux/fs.h> // BLKGETSIZE64
|
||||||
|
#elif defined __FreeBSD__
|
||||||
|
#include <sys/disk.h> // DIOCGMEDIASIZE
|
||||||
|
#elif defined __APPLE__
|
||||||
|
#include <sys/disk.h> // DKIOCGETBLOCKCOUNT / DKIOCGETBLOCKSIZE
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace DiscIO
|
namespace DiscIO
|
||||||
|
@ -25,24 +35,39 @@ namespace DiscIO
|
||||||
|
|
||||||
DriveReader::DriveReader(const std::string& drive)
|
DriveReader::DriveReader(const std::string& drive)
|
||||||
{
|
{
|
||||||
|
// 32 sectors is roughly the optimal amount a CD Drive can read in
|
||||||
|
// a single IO cycle. Larger values yield no performance improvement
|
||||||
|
// and just cause IO stalls from the read delay. Smaller values allow
|
||||||
|
// the OS IO and seeking overhead to ourstrip the time actually spent
|
||||||
|
// transferring bytes from the media.
|
||||||
|
SetChunkSize(32); // 32*2048 = 64KiB
|
||||||
|
SetSectorSize(2048);
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
SectorReader::SetSectorSize(2048);
|
|
||||||
auto const path = UTF8ToTStr(std::string("\\\\.\\") + drive);
|
auto const path = UTF8ToTStr(std::string("\\\\.\\") + drive);
|
||||||
m_disc_handle = CreateFile(path.c_str(), GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE,
|
m_disc_handle = CreateFile(path.c_str(), GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE,
|
||||||
nullptr, OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS, nullptr);
|
nullptr, OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS, nullptr);
|
||||||
if (m_disc_handle != INVALID_HANDLE_VALUE)
|
if (IsOK())
|
||||||
{
|
{
|
||||||
// Do a test read to make sure everything is OK, since it seems you can get
|
// Do a test read to make sure everything is OK, since it seems you can get
|
||||||
// handles to empty drives.
|
// handles to empty drives.
|
||||||
DWORD not_used;
|
DWORD not_used;
|
||||||
std::vector<u8> buffer(m_blocksize);
|
std::vector<u8> buffer(GetSectorSize());
|
||||||
if (!ReadFile(m_disc_handle, buffer.data(), m_blocksize, ¬_used, nullptr))
|
if (!ReadFile(m_disc_handle, buffer.data(), GetSectorSize(), ¬_used, nullptr))
|
||||||
{
|
{
|
||||||
// OK, something is wrong.
|
// OK, something is wrong.
|
||||||
CloseHandle(m_disc_handle);
|
CloseHandle(m_disc_handle);
|
||||||
m_disc_handle = INVALID_HANDLE_VALUE;
|
m_disc_handle = INVALID_HANDLE_VALUE;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if (IsOK())
|
||||||
|
{
|
||||||
|
// Initialize m_size by querying the volume capacity.
|
||||||
|
STORAGE_READ_CAPACITY storage_size;
|
||||||
|
storage_size.Version = sizeof(storage_size);
|
||||||
|
DWORD bytes = 0;
|
||||||
|
DeviceIoControl(m_disc_handle, IOCTL_STORAGE_READ_CAPACITY, nullptr, 0,
|
||||||
|
&storage_size, sizeof(storage_size), &bytes, nullptr);
|
||||||
|
m_size = bytes ? storage_size.DiskLength.QuadPart : 0;
|
||||||
|
|
||||||
#ifdef _LOCKDRIVE // Do we want to lock the drive?
|
#ifdef _LOCKDRIVE // Do we want to lock the drive?
|
||||||
// Lock the compact disc in the CD-ROM drive to prevent accidental
|
// Lock the compact disc in the CD-ROM drive to prevent accidental
|
||||||
|
@ -53,10 +78,24 @@ DriveReader::DriveReader(const std::string& drive)
|
||||||
0, &dwNotUsed, nullptr);
|
0, &dwNotUsed, nullptr);
|
||||||
#endif
|
#endif
|
||||||
#else
|
#else
|
||||||
SectorReader::SetSectorSize(2048);
|
|
||||||
m_file.Open(drive, "rb");
|
m_file.Open(drive, "rb");
|
||||||
if (m_file)
|
if (m_file)
|
||||||
{
|
{
|
||||||
|
int fd = fileno(m_file.GetHandle());
|
||||||
|
#if defined __linux__
|
||||||
|
// NOTE: Doesn't matter if it fails, m_size was initialized to zero
|
||||||
|
ioctl(fd, BLKGETSIZE64, &m_size); // u64*
|
||||||
|
#elif defined __FreeBSD__
|
||||||
|
off_t size = 0;
|
||||||
|
ioctl(fd, DIOCGMEDIASIZE, &size); // off_t*
|
||||||
|
m_size = size;
|
||||||
|
#elif defined __APPLE__
|
||||||
|
u64 count = 0;
|
||||||
|
u32 block_size = 0;
|
||||||
|
ioctl(fd, DKIOCGETBLOCKCOUNT, &count); // u64*
|
||||||
|
ioctl(fd, DKIOCGETBLOCKSIZE, &block_size); // u32*
|
||||||
|
m_size = count * block_size;
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -95,45 +134,32 @@ std::unique_ptr<DriveReader> DriveReader::Create(const std::string& drive)
|
||||||
return reader;
|
return reader;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DriveReader::GetBlock(u64 block_num, u8* out_ptr)
|
bool DriveReader::GetBlock(u64 block_num, u8* out_ptr)
|
||||||
{
|
{
|
||||||
std::vector<u8> sector(m_blocksize);
|
return DriveReader::ReadMultipleAlignedBlocks(block_num, 1, out_ptr);
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
u64 offset = m_blocksize * block_num;
|
|
||||||
LONG off_low = (LONG)offset & 0xFFFFFFFF;
|
|
||||||
LONG off_high = (LONG)(offset >> 32);
|
|
||||||
DWORD not_used;
|
|
||||||
SetFilePointer(m_disc_handle, off_low, &off_high, FILE_BEGIN);
|
|
||||||
if (!ReadFile(m_disc_handle, sector.data(), m_blocksize, ¬_used, nullptr))
|
|
||||||
PanicAlertT("Disc Read Error");
|
|
||||||
#else
|
|
||||||
m_file.Seek(m_blocksize * block_num, SEEK_SET);
|
|
||||||
m_file.ReadBytes(sector.data(), m_blocksize);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
std::copy(sector.begin(), sector.end(), out_ptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DriveReader::ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr)
|
bool DriveReader::ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
u64 offset = m_blocksize * block_num;
|
LARGE_INTEGER offset;
|
||||||
LONG off_low = (LONG)offset & 0xFFFFFFFF;
|
offset.QuadPart = GetSectorSize() * block_num;
|
||||||
LONG off_high = (LONG)(offset >> 32);
|
SetFilePointerEx(m_disc_handle, offset, nullptr, FILE_BEGIN);
|
||||||
DWORD not_used;
|
DWORD bytes_read;
|
||||||
SetFilePointer(m_disc_handle, off_low, &off_high, FILE_BEGIN);
|
if (!ReadFile(m_disc_handle, out_ptr, static_cast<DWORD>(GetSectorSize() * num_blocks),
|
||||||
if (!ReadFile(m_disc_handle, out_ptr, (DWORD)(m_blocksize * num_blocks), ¬_used, nullptr))
|
&bytes_read, nullptr))
|
||||||
{
|
{
|
||||||
PanicAlertT("Disc Read Error");
|
PanicAlertT("Disc Read Error");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
return bytes_read == GetSectorSize() * num_blocks;
|
||||||
#else
|
#else
|
||||||
fseeko(m_file.GetHandle(), (m_blocksize * block_num), SEEK_SET);
|
m_file.Seek(GetSectorSize() * block_num, SEEK_SET);
|
||||||
if (fread(out_ptr, 1, (m_blocksize * num_blocks), m_file.GetHandle()) != (m_blocksize * num_blocks))
|
if (m_file.ReadBytes(out_ptr, num_blocks * GetSectorSize()))
|
||||||
return false;
|
return true;
|
||||||
|
m_file.Clear();
|
||||||
|
return false;
|
||||||
#endif
|
#endif
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
|
@ -30,18 +30,18 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DriveReader(const std::string& drive);
|
DriveReader(const std::string& drive);
|
||||||
void GetBlock(u64 block_num, u8 *out_ptr) override;
|
bool GetBlock(u64 block_num, u8 *out_ptr) override;
|
||||||
bool ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr) override;
|
bool ReadMultipleAlignedBlocks(u64 block_num, u64 num_blocks, u8* out_ptr) override;
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
HANDLE m_disc_handle;
|
HANDLE m_disc_handle = INVALID_HANDLE_VALUE;
|
||||||
PREVENT_MEDIA_REMOVAL m_lock_cdrom;
|
PREVENT_MEDIA_REMOVAL m_lock_cdrom;
|
||||||
bool IsOK() { return m_disc_handle != INVALID_HANDLE_VALUE; }
|
bool IsOK() const { return m_disc_handle != INVALID_HANDLE_VALUE; }
|
||||||
#else
|
#else
|
||||||
File::IOFile m_file;
|
File::IOFile m_file;
|
||||||
bool IsOK() { return m_file != nullptr; }
|
bool IsOK() const { return m_file.IsOpen() && m_file.IsGood(); }
|
||||||
#endif
|
#endif
|
||||||
s64 m_size;
|
u64 m_size = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cinttypes>
|
#include <cinttypes>
|
||||||
|
#include <cmath>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
@ -345,24 +346,22 @@ void CGameListCtrl::Update()
|
||||||
SetFocus();
|
SetFocus();
|
||||||
}
|
}
|
||||||
|
|
||||||
static wxString NiceSizeFormat(u64 _size)
|
static wxString NiceSizeFormat(u64 size)
|
||||||
{
|
{
|
||||||
// Return a pretty filesize string from byte count.
|
// Return a pretty filesize string from byte count.
|
||||||
// e.g. 1134278 -> "1.08 MiB"
|
// e.g. 1134278 -> "1.08 MiB"
|
||||||
|
|
||||||
const char* const unit_symbols[] = {"B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"};
|
const char* const unit_symbols[] = { "B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB" };
|
||||||
|
|
||||||
// Find largest power of 2 less than _size.
|
// Find largest power of 2 less than size.
|
||||||
// div 10 to get largest named unit less than _size
|
// div 10 to get largest named unit less than size
|
||||||
// 10 == log2(1024) (number of B in a KiB, KiB in a MiB, etc)
|
// 10 == log2(1024) (number of B in a KiB, KiB in a MiB, etc)
|
||||||
const u64 unit = IntLog2(std::max<u64>(_size, 1)) / 10;
|
// Max value is 63 / 10 = 6
|
||||||
const u64 unit_size = (1ull << (unit * 10));
|
const int unit = IntLog2(std::max<u64>(size, 1)) / 10;
|
||||||
|
|
||||||
// mul 1000 for 3 decimal places, add 5 to round up, div 10 for 2 decimal places
|
// Don't need exact values, only 5 most significant digits
|
||||||
std::string value = std::to_string((_size * 1000 / unit_size + 5) / 10);
|
double unit_size = std::pow(2, unit * 10);
|
||||||
// Insert decimal point.
|
return wxString::Format("%.2f %s", size / unit_size, unit_symbols[unit]);
|
||||||
value.insert(value.size() - 2, ".");
|
|
||||||
return StrToWxStr(StringFromFormat("%s %s", value.c_str(), unit_symbols[unit]));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the column content of the item at _Index
|
// Update the column content of the item at _Index
|
||||||
|
|
|
@ -1501,8 +1501,11 @@ void CISOProperties::ChangeBannerDetails(DiscIO::IVolume::ELanguage language)
|
||||||
m_Comment->SetValue(comment);
|
m_Comment->SetValue(comment);
|
||||||
m_Maker->SetValue(maker);//dev too
|
m_Maker->SetValue(maker);//dev too
|
||||||
|
|
||||||
std::string filename, extension;
|
std::string path, filename, extension;
|
||||||
SplitPath(OpenGameListItem.GetFileName(), nullptr, &filename, &extension);
|
SplitPath(OpenGameListItem.GetFileName(), &path, &filename, &extension);
|
||||||
|
// Real disk drives don't have filenames on Windows
|
||||||
|
if (filename.empty() && extension.empty())
|
||||||
|
filename = path + ' ';
|
||||||
// Also sets the window's title
|
// Also sets the window's title
|
||||||
SetTitle(StrToWxStr(StringFromFormat("%s%s: %s - ", filename.c_str(),
|
SetTitle(StrToWxStr(StringFromFormat("%s%s: %s - ", filename.c_str(),
|
||||||
extension.c_str(), OpenGameListItem.GetUniqueID().c_str())) + name);
|
extension.c_str(), OpenGameListItem.GetUniqueID().c_str())) + name);
|
||||||
|
|
Loading…
Reference in New Issue