CDVD: Make CSO file reader threaded

This commit is contained in:
TellowKrinkle 2021-03-24 04:21:21 -05:00 committed by lightningterror
parent bdcfcc65ea
commit ae945f49e3
2 changed files with 48 additions and 163 deletions

View File

@ -85,9 +85,9 @@ bool CsoFileReader::ValidateHeader(const CsoHeader& hdr)
return true;
}
bool CsoFileReader::Open(const wxString& fileName)
bool CsoFileReader::Open2(const wxString& fileName)
{
Close();
Close2();
m_filename = fileName;
m_src = PX_fopen_rb(m_filename);
@ -99,7 +99,7 @@ bool CsoFileReader::Open(const wxString& fileName)
if (!success)
{
Close();
Close2();
return false;
}
return true;
@ -152,10 +152,6 @@ bool CsoFileReader::InitializeBuffers()
m_readBuffer = new u8[m_frameSize + (1 << m_indexShift)];
}
// This is a buffer for the most recently decompressed frame.
m_zlibBuffer = new u8[m_frameSize + (1 << m_indexShift)];
m_zlibBufferFrame = numFrames;
const u32 indexSize = numFrames + 1;
m_index = new u32[indexSize];
if (fread(m_index, sizeof(u32), indexSize, m_src) != indexSize)
@ -177,12 +173,9 @@ bool CsoFileReader::InitializeBuffers()
return true;
}
void CsoFileReader::Close()
void CsoFileReader::Close2()
{
m_filename.Empty();
#if CSO_USE_CHUNKSCACHE
m_cache.Clear();
#endif
if (m_src)
{
@ -200,11 +193,6 @@ void CsoFileReader::Close()
delete[] m_readBuffer;
m_readBuffer = NULL;
}
if (m_zlibBuffer)
{
delete[] m_zlibBuffer;
m_zlibBuffer = NULL;
}
if (m_index)
{
delete[] m_index;
@ -212,68 +200,28 @@ void CsoFileReader::Close()
}
}
int CsoFileReader::ReadSync(void* pBuffer, uint sector, uint count)
ThreadedFileReader::Chunk CsoFileReader::ChunkForOffset(u64 offset)
{
if (!m_src)
Chunk chunk = {0};
if (offset >= m_totalSize)
{
return 0;
chunk.chunkID = -1;
}
// Note that, in practice, count will always be 1. It seems one sector is read
// per interrupt, even if multiple are requested by the application.
u8* dest = (u8*)pBuffer;
// We do it this way in case m_blocksize is not well aligned to our frame size.
u64 pos = (u64)sector * (u64)m_blocksize;
int remaining = count * m_blocksize;
int bytes = 0;
while (remaining > 0)
else
{
int readBytes;
#if CSO_USE_CHUNKSCACHE
// Try first to read from the cache.
readBytes = m_cache.Read(dest + bytes, pos + bytes, remaining);
#else
readBytes = -1;
#endif
if (readBytes < 0)
{
readBytes = ReadFromFrame(dest + bytes, pos + bytes, remaining);
if (readBytes == 0)
{
// We hit EOF.
break;
}
#if CSO_USE_CHUNKSCACHE
// Add the bytes into the cache. We need to allocate a buffer for it.
void* cached = malloc(readBytes);
memcpy(cached, dest + bytes, readBytes);
m_cache.Take(cached, pos + bytes, readBytes, readBytes);
#endif
}
bytes += readBytes;
remaining -= readBytes;
chunk.chunkID = offset >> m_frameShift;
chunk.length = m_frameSize;
chunk.offset = chunk.chunkID << m_frameShift;
}
return bytes;
return chunk;
}
int CsoFileReader::ReadFromFrame(u8* dest, u64 pos, int maxBytes)
int CsoFileReader::ReadChunk(void *dst, s64 chunkID)
{
if (pos >= m_totalSize)
{
// Can't read anything passed the end.
return 0;
}
if (chunkID < 0)
return -1;
const u32 frame = (u32)(pos >> m_frameShift);
const u32 offset = (u32)(pos - (frame << m_frameShift));
// This is how many bytes we will actually be reading from this frame.
const u32 bytes = (u32)(std::min(m_blocksize, static_cast<uint>(m_frameSize - offset)));
const u32 frame = chunkID;
// Grab the index data for the frame we're about to read.
const bool compressed = (m_index[frame + 0] & 0x80000000) == 0;
@ -287,77 +235,36 @@ int CsoFileReader::ReadFromFrame(u8* dest, u64 pos, int maxBytes)
if (!compressed)
{
// Just read directly, easy.
if (PX_fseeko(m_src, m_dataoffset + frameRawPos + offset, SEEK_SET) != 0)
if (PX_fseeko(m_src, frameRawPos, SEEK_SET) != 0)
{
Console.Error("Unable to seek to uncompressed CSO data.");
return 0;
}
return fread(dest, 1, bytes, m_src);
return fread(dst, 1, m_frameSize, m_src);
}
else
{
// We don't need to decompress if we already did this same frame last time.
if (m_zlibBufferFrame != frame)
if (PX_fseeko(m_src, frameRawPos, SEEK_SET) != 0)
{
if (PX_fseeko(m_src, m_dataoffset + frameRawPos, SEEK_SET) != 0)
{
Console.Error("Unable to seek to compressed CSO data.");
return 0;
}
// This might be less bytes than frameRawSize in case of padding on the last frame.
// This is because the index positions must be aligned.
const u32 readRawBytes = fread(m_readBuffer, 1, frameRawSize, m_src);
if (!DecompressFrame(frame, readRawBytes))
{
return 0;
}
Console.Error("Unable to seek to compressed CSO data.");
return 0;
}
// This might be less bytes than frameRawSize in case of padding on the last frame.
// This is because the index positions must be aligned.
const u32 readRawBytes = fread(m_readBuffer, 1, frameRawSize, m_src);
// Now we just copy the offset data from the cache.
memcpy(dest, m_zlibBuffer + offset, bytes);
m_z_stream->next_in = m_readBuffer;
m_z_stream->avail_in = readRawBytes;
m_z_stream->next_out = static_cast<Bytef*>(dst);
m_z_stream->avail_out = m_frameSize;
int status = inflate(m_z_stream, Z_FINISH);
bool success = status == Z_STREAM_END && m_z_stream->total_out == m_frameSize;
if (!success)
Console.Error("Unable to decompress CSO frame using zlib.");
inflateReset(m_z_stream);
return success ? m_frameSize : 0;
}
return bytes;
}
bool CsoFileReader::DecompressFrame(u32 frame, u32 readBufferSize)
{
m_z_stream->next_in = m_readBuffer;
m_z_stream->avail_in = readBufferSize;
m_z_stream->next_out = m_zlibBuffer;
m_z_stream->avail_out = m_frameSize;
int status = inflate(m_z_stream, Z_FINISH);
bool success = status == Z_STREAM_END && m_z_stream->total_out == m_frameSize;
if (success)
{
// Our buffer now contains this frame.
m_zlibBufferFrame = frame;
}
else
{
Console.Error("Unable to decompress CSO frame using zlib.");
m_zlibBufferFrame = (u32)-1;
}
inflateReset(m_z_stream);
return success;
}
void CsoFileReader::BeginRead(void* pBuffer, uint sector, uint count)
{
// TODO: No async support yet, implement as sync.
m_bytesRead = ReadSync(pBuffer, sector, count);
}
int CsoFileReader::FinishRead()
{
int res = m_bytesRead;
m_bytesRead = -1;
return res;
}
void CsoFileReader::CancelRead()
{
// TODO: No async read support yet.
}

View File

@ -24,7 +24,7 @@
// For this reason, it's currently disabled.
#define CSO_USE_CHUNKSCACHE 0
#include "AsyncFileReader.h"
#include "ThreadedFileReader.h"
#include "ChunksCache.h"
struct CsoHeader;
@ -32,7 +32,7 @@ typedef struct z_stream_s z_stream;
static const uint CSO_CHUNKCACHE_SIZE_MB = 200;
class CsoFileReader : public AsyncFileReader
class CsoFileReader : public ThreadedFileReader
{
DeclareNoncopyableObject(CsoFileReader);
@ -42,66 +42,44 @@ public:
, m_frameShift(0)
, m_indexShift(0)
, m_readBuffer(0)
, m_zlibBuffer(0)
, m_zlibBufferFrame(0)
, m_index(0)
, m_totalSize(0)
, m_src(0)
, m_z_stream(0)
,
#if CSO_USE_CHUNKSCACHE
m_cache(CSO_CHUNKCACHE_SIZE_MB)
,
#endif
m_bytesRead(0)
{
m_blocksize = 2048;
};
virtual ~CsoFileReader(void) { Close(); };
~CsoFileReader(void) { Close(); };
static bool CanHandle(const wxString& fileName);
virtual bool Open(const wxString& fileName);
bool Open2(const wxString& fileName) override;
virtual int ReadSync(void* pBuffer, uint sector, uint count);
Chunk ChunkForOffset(u64 offset) override;
int ReadChunk(void *dst, s64 chunkID) override;
virtual void BeginRead(void* pBuffer, uint sector, uint count);
virtual int FinishRead(void);
virtual void CancelRead(void);
void Close2(void) override;
virtual void Close(void);
virtual uint GetBlockCount(void) const
uint GetBlockCount(void) const override
{
return (m_totalSize - m_dataoffset) / m_blocksize;
};
virtual void SetBlockSize(uint bytes) { m_blocksize = bytes; }
virtual void SetDataOffset(int bytes) { m_dataoffset = bytes; }
private:
static bool ValidateHeader(const CsoHeader& hdr);
bool ReadFileHeader();
bool InitializeBuffers();
int ReadFromFrame(u8* dest, u64 pos, int maxBytes);
bool DecompressFrame(Bytef* dst, u32 frame, u32 readBufferSize);
bool DecompressFrame(u32 frame, u32 readBufferSize);
u32 m_frameSize;
u8 m_frameShift;
u8 m_indexShift;
u8* m_readBuffer;
u8* m_zlibBuffer;
u32 m_zlibBufferFrame;
u8* m_readBuffer;;
u32* m_index;
u64 m_totalSize;
// The actual source cso file handle.
FILE* m_src;
z_stream* m_z_stream;
#if CSO_USE_CHUNKSCACHE
ChunksCache m_cache;
#endif
// The result of a read is stored here between BeginRead() and FinishRead().
int m_bytesRead;
};