mirror of
https://github.com/isledecomp/isle.git
synced 2024-11-23 08:08:03 -05:00
51ec2c97c6
* 100% Match of MxDSFile * ...almost, MxDSFile::Open is still not quite matching but all of the other methods are 100% matching. * Turns out that most of the virtual methods and some of the members are actually on the MxDSSource base class, which I've pulled out as part of this. * In order to implement the methods I added the MXIOINFO class, which seems to be a thin wrapper around the MMIOINFO windows structure. We can tell this because MMIOINFO::~MMIOINFO was included in the DLL exports, and calls down to a function which deconstructs something looking exactly like MMIOINFO. * Add mxdssource.cpp * mattkc feedback * some accuracy improvements * Use FOURCC macro * Tirival solve in mxioinfo.cpp * Update mxdsfile.cpp 0xFFFFFFFF -> -1 --------- Co-authored-by: Christian Semmler <mail@csemmler.com>
130 lines
2.8 KiB
C++
130 lines
2.8 KiB
C++
#include "mxdsfile.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#define SI_MAJOR_VERSION 2
|
|
#define SI_MINOR_VERSION 2
|
|
|
|
#define FOURCC(a, b, c, d) (((a) << 0) | ((b) << 8) | ((c) << 16) | ((d) << 24))
|
|
|
|
// OFFSET: LEGO1 0x100cc4b0
|
|
MxDSFile::MxDSFile(const char *filename, unsigned long skipReadingChunks)
|
|
{
|
|
m_filename = filename;
|
|
m_skipReadingChunks = skipReadingChunks;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100bfed0
|
|
MxDSFile::~MxDSFile()
|
|
{
|
|
Close();
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc590
|
|
long MxDSFile::Open(unsigned long uStyle)
|
|
{
|
|
// No idea what's stopping this one matching, but I'm pretty
|
|
// confident it has the correct behavior.
|
|
long longResult = 1;
|
|
memset(&m_io, 0, sizeof(MXIOINFO));
|
|
|
|
if (m_io.Open(m_filename.GetData(), uStyle) != 0) {
|
|
return -1;
|
|
}
|
|
|
|
m_io.SetBuffer(NULL, 0, 0);
|
|
m_position = 0;
|
|
|
|
if (m_skipReadingChunks == 0) {
|
|
longResult = ReadChunks();
|
|
}
|
|
|
|
if (longResult != 0) {
|
|
Close(); // vtable + 0x18
|
|
}
|
|
else {
|
|
Seek(0, 0); // vtable + 0x24
|
|
}
|
|
|
|
return longResult;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc780
|
|
long MxDSFile::Read(unsigned char *pch, unsigned long cch)
|
|
{
|
|
if (m_io.Read((char*)pch, cch) != cch)
|
|
return -1;
|
|
|
|
m_position += cch;
|
|
return 0;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc620
|
|
long MxDSFile::ReadChunks()
|
|
{
|
|
_MMCKINFO topChunk;
|
|
_MMCKINFO childChunk;
|
|
char tempBuffer[80];
|
|
|
|
topChunk.fccType = FOURCC('O', 'M', 'N', 'I');
|
|
if (m_io.Descend(&topChunk, NULL, MMIO_FINDRIFF) != 0) {
|
|
return -1;
|
|
}
|
|
childChunk.ckid = FOURCC('M', 'x', 'H', 'd');
|
|
if (m_io.Descend(&childChunk, &topChunk, 0) != 0) {
|
|
return -1;
|
|
}
|
|
|
|
m_io.Read((char*)&m_header, 0xc);
|
|
if ((m_header.majorVersion == SI_MAJOR_VERSION) && (m_header.minorVersion == SI_MINOR_VERSION))
|
|
{
|
|
childChunk.ckid = FOURCC('M', 'x', 'O', 'f');
|
|
if (m_io.Descend(&childChunk, &topChunk, 0) != 0) {
|
|
return -1;
|
|
}
|
|
unsigned long* pLengthInDWords = &m_lengthInDWords;
|
|
m_io.Read((char *)pLengthInDWords, 4);
|
|
m_pBuffer = malloc(*pLengthInDWords * 4);
|
|
m_io.Read((char*)m_pBuffer, *pLengthInDWords * 4);
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
sprintf(tempBuffer, "Wrong SI file version. %d.%d expected.", SI_MAJOR_VERSION, SI_MINOR_VERSION);
|
|
MessageBoxA(NULL, tempBuffer, NULL, MB_ICONERROR);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc7b0
|
|
long MxDSFile::Seek(long lOffset, int iOrigin)
|
|
{
|
|
return (m_position = m_io.Seek(lOffset, iOrigin)) == -1 ? -1 : 0;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc7e0
|
|
unsigned long MxDSFile::GetBufferSize()
|
|
{
|
|
return m_header.bufferSize;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc7f0
|
|
unsigned long MxDSFile::GetStreamBuffersNum()
|
|
{
|
|
return m_header.streamBuffersNum;
|
|
}
|
|
|
|
// OFFSET: LEGO1 0x100cc740
|
|
long MxDSFile::Close()
|
|
{
|
|
m_io.Close(0);
|
|
m_position = -1;
|
|
memset(&m_header, 0, sizeof(m_header));
|
|
if (m_lengthInDWords != 0)
|
|
{
|
|
m_lengthInDWords = 0;
|
|
free(m_pBuffer);
|
|
m_pBuffer = NULL;
|
|
}
|
|
return 0;
|
|
}
|