#include "stdafx.h"
#include "Loaders.h"
#include "UMXTools.h"
OPENMPT_NAMESPACE_BEGIN
CSoundFile::ProbeResult CSoundFile::ProbeFileHeaderUAX(MemoryFileReader file, const uint64 *pfilesize)
{
UMXFileHeader fileHeader;
if(!file.ReadStruct(fileHeader))
{
return ProbeWantMoreData;
}
if(!fileHeader.IsValid())
{
return ProbeFailure;
}
if(!FindUMXNameTableEntryMemory(file, fileHeader, "sound"))
{
return ProbeFailure;
}
MPT_UNREFERENCED_PARAMETER(pfilesize);
return ProbeSuccess;
}
bool CSoundFile::ReadUAX(FileReader &file, ModLoadingFlags loadFlags)
{
file.Rewind();
UMXFileHeader fileHeader;
if(!file.ReadStruct(fileHeader)
|| !fileHeader.IsValid())
{
return false;
}
if(!FindUMXNameTableEntry(file, fileHeader, "sound"))
{
return false;
}
if(loadFlags == onlyVerifyHeader)
{
return true;
}
std::vector<std::string> names = ReadUMXNameTable(file, fileHeader);
if(!file.Seek(fileHeader.importOffset))
{
return false;
}
std::vector<int32> classes;
classes.reserve(fileHeader.importCount);
for(uint32 i = 0; i < fileHeader.importCount && file.CanRead(4); i++)
{
int32 objName = ReadUMXImportTableEntry(file, fileHeader.packageVersion);
if(static_cast<size_t>(objName) < names.size())
{
classes.push_back(objName);
}
}
if(!file.Seek(fileHeader.exportOffset))
{
return false;
}
InitializeGlobals();
m_modFormat.formatName = mpt::format(U_("Unreal Package v%1"))(fileHeader.packageVersion);
m_modFormat.type = U_("uax");
m_modFormat.charset = mpt::CharsetWindows1252;
for(uint32 i = 0; i < fileHeader.exportCount && file.CanRead(4); i++)
{
int32 objClass, objOffset, objSize, objName;
ReadUMXExportTableEntry(file, objClass, objOffset, objSize, objName, fileHeader.packageVersion);
if(objSize <= 0 || objClass >= 0)
{
continue;
}
objClass = -objClass - 1;
bool isSound = false;
if(static_cast<size_t>(objClass) < classes.size())
{
isSound = (names[classes[objClass]] == "sound");
}
if(!isSound)
{
continue;
}
FileReader chunk = file.GetChunkAt(objOffset, objSize);
if(chunk.IsValid())
{
if(fileHeader.packageVersion < 40)
{
chunk.Skip(8); }
if(fileHeader.packageVersion < 60)
{
chunk.Skip(16); }
#if 0#else
ReadUMXIndex(chunk);
#endif
if(fileHeader.packageVersion >= 120)
{
ReadUMXIndex(chunk);
chunk.Skip(8);
} else if(fileHeader.packageVersion >= 100)
{
chunk.Skip(4);
ReadUMXIndex(chunk);
chunk.Skip(4);
} else if(fileHeader.packageVersion >= 62)
{
ReadUMXIndex(chunk);
chunk.Skip(4);
} else
{
ReadUMXIndex(chunk);
}
int32 size = ReadUMXIndex(chunk);
FileReader fileChunk = chunk.ReadChunk(size);
if(GetNumSamples() < MAX_SAMPLES - 1)
{
if(ReadSampleFromFile(GetNumSamples() + 1, fileChunk, true))
{
if(static_cast<size_t>(objName) < names.size())
{
mpt::String::Copy(m_szNames[GetNumSamples()], names[objName]);
}
}
}
}
}
if(m_nSamples != 0)
{
InitializeChannels();
SetType(MOD_TYPE_MPT);
m_ContainerType = MOD_CONTAINERTYPE_UAX;
m_nChannels = 4;
Patterns.Insert(0, 64);
Order().assign(1, 0);
return true;
} else
{
return false;
}
}
OPENMPT_NAMESPACE_END