mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-18 23:21:41 +00:00
- added a method to FileReader to read the contents into an array and used it on the MIDI sources for testing.
This commit is contained in:
parent
c07aeb7498
commit
1ccbbcb81d
5 changed files with 62 additions and 71 deletions
|
@ -184,6 +184,14 @@ public:
|
||||||
return mReader->Read(buffer, (long)len);
|
return mReader->Read(buffer, (long)len);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TArray<uint8_t> Read()
|
||||||
|
{
|
||||||
|
TArray<uint8_t> buffer(mReader->Length, true);
|
||||||
|
Size length = mReader->Read(&buffer[0], mReader->Length);
|
||||||
|
if (length < mReader->Length) buffer.Clear();
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
char *Gets(char *strbuf, Size len)
|
char *Gets(char *strbuf, Size len)
|
||||||
{
|
{
|
||||||
return mReader->Gets(strbuf, (int)len);
|
return mReader->Gets(strbuf, (int)len);
|
||||||
|
|
|
@ -121,8 +121,7 @@ private:
|
||||||
uint32_t *SendCommand (uint32_t *event, TrackInfo *track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
|
uint32_t *SendCommand (uint32_t *event, TrackInfo *track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
|
||||||
TrackInfo *FindNextDue ();
|
TrackInfo *FindNextDue ();
|
||||||
|
|
||||||
uint8_t *MusHeader;
|
TArray<uint8_t> MusHeader;
|
||||||
int SongLen;
|
|
||||||
TrackInfo *Tracks;
|
TrackInfo *Tracks;
|
||||||
TrackInfo *TrackDue;
|
TrackInfo *TrackDue;
|
||||||
int NumTracks;
|
int NumTracks;
|
||||||
|
@ -181,8 +180,7 @@ private:
|
||||||
static uint32_t ReadVarLenHMI(TrackInfo *);
|
static uint32_t ReadVarLenHMI(TrackInfo *);
|
||||||
static uint32_t ReadVarLenHMP(TrackInfo *);
|
static uint32_t ReadVarLenHMP(TrackInfo *);
|
||||||
|
|
||||||
uint8_t *MusHeader;
|
TArray<uint8_t> MusHeader;
|
||||||
int SongLen;
|
|
||||||
int NumTracks;
|
int NumTracks;
|
||||||
TrackInfo *Tracks;
|
TrackInfo *Tracks;
|
||||||
TrackInfo *TrackDue;
|
TrackInfo *TrackDue;
|
||||||
|
@ -218,8 +216,7 @@ private:
|
||||||
uint32_t *SendCommand (uint32_t *event, EventSource track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
|
uint32_t *SendCommand (uint32_t *event, EventSource track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
|
||||||
EventSource FindNextDue();
|
EventSource FindNextDue();
|
||||||
|
|
||||||
uint8_t *MusHeader;
|
TArray<uint8_t> MusHeader;
|
||||||
int SongLen; // length of the entire file
|
|
||||||
int NumSongs;
|
int NumSongs;
|
||||||
TrackInfo *Songs;
|
TrackInfo *Songs;
|
||||||
TrackInfo *CurrSong;
|
TrackInfo *CurrSong;
|
||||||
|
|
|
@ -129,19 +129,17 @@ HMISong::HMISong (FileReader &reader)
|
||||||
{ // Way too small to be HMI.
|
{ // Way too small to be HMI.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
MusHeader = new uint8_t[len];
|
MusHeader = reader.Read();
|
||||||
SongLen = len;
|
|
||||||
NumTracks = 0;
|
NumTracks = 0;
|
||||||
if (reader.Read(MusHeader, len) != len)
|
if (MusHeader.Size() == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Do some validation of the MIDI file
|
// Do some validation of the MIDI file
|
||||||
if (memcmp(MusHeader, HMI_SONG_MAGIC, sizeof(HMI_SONG_MAGIC)) == 0)
|
if (memcmp(&MusHeader[0], HMI_SONG_MAGIC, sizeof(HMI_SONG_MAGIC)) == 0)
|
||||||
{
|
{
|
||||||
SetupForHMI(len);
|
SetupForHMI(len);
|
||||||
}
|
}
|
||||||
else if (((uint32_t *)MusHeader)[0] == MAKE_ID('H','M','I','M') &&
|
else if (memcmp(&MusHeader[0], "HMIMIDIP", 8) == 0)
|
||||||
((uint32_t *)MusHeader)[1] == MAKE_ID('I','D','I','P'))
|
|
||||||
{
|
{
|
||||||
SetupForHMP(len);
|
SetupForHMP(len);
|
||||||
}
|
}
|
||||||
|
@ -155,14 +153,10 @@ HMISong::HMISong (FileReader &reader)
|
||||||
|
|
||||||
HMISong::~HMISong()
|
HMISong::~HMISong()
|
||||||
{
|
{
|
||||||
if (Tracks != NULL)
|
if (Tracks != nullptr)
|
||||||
{
|
{
|
||||||
delete[] Tracks;
|
delete[] Tracks;
|
||||||
}
|
}
|
||||||
if (MusHeader != NULL)
|
|
||||||
{
|
|
||||||
delete[] MusHeader;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -175,8 +169,10 @@ void HMISong::SetupForHMI(int len)
|
||||||
{
|
{
|
||||||
int i, p;
|
int i, p;
|
||||||
|
|
||||||
|
auto MusPtr = &MusHeader[0];
|
||||||
|
|
||||||
ReadVarLen = ReadVarLenHMI;
|
ReadVarLen = ReadVarLenHMI;
|
||||||
NumTracks = GetShort(MusHeader + HMI_TRACK_COUNT_OFFSET);
|
NumTracks = GetShort(MusPtr + HMI_TRACK_COUNT_OFFSET);
|
||||||
|
|
||||||
if (NumTracks <= 0)
|
if (NumTracks <= 0)
|
||||||
{
|
{
|
||||||
|
@ -187,16 +183,16 @@ void HMISong::SetupForHMI(int len)
|
||||||
// HMI files have two values here, a full value and a quarter value. Some games,
|
// HMI files have two values here, a full value and a quarter value. Some games,
|
||||||
// notably Quarantines, have identical values for some reason, so it's safer to
|
// notably Quarantines, have identical values for some reason, so it's safer to
|
||||||
// use the quarter value and multiply it by four than to trust the full value.
|
// use the quarter value and multiply it by four than to trust the full value.
|
||||||
Division = GetShort(MusHeader + HMI_DIVISION_OFFSET) << 2;
|
Division = GetShort(MusPtr + HMI_DIVISION_OFFSET) << 2;
|
||||||
Tempo = InitialTempo = 4000000;
|
Tempo = InitialTempo = 4000000;
|
||||||
|
|
||||||
Tracks = new TrackInfo[NumTracks + 1];
|
Tracks = new TrackInfo[NumTracks + 1];
|
||||||
int track_dir = GetInt(MusHeader + HMI_TRACK_DIR_PTR_OFFSET);
|
int track_dir = GetInt(MusPtr + HMI_TRACK_DIR_PTR_OFFSET);
|
||||||
|
|
||||||
// Gather information about each track
|
// Gather information about each track
|
||||||
for (i = 0, p = 0; i < NumTracks; ++i)
|
for (i = 0, p = 0; i < NumTracks; ++i)
|
||||||
{
|
{
|
||||||
int start = GetInt(MusHeader + track_dir + i*4);
|
int start = GetInt(MusPtr + track_dir + i*4);
|
||||||
int tracklen, datastart;
|
int tracklen, datastart;
|
||||||
|
|
||||||
if (start > len - HMITRACK_DESIGNATION_OFFSET - 4)
|
if (start > len - HMITRACK_DESIGNATION_OFFSET - 4)
|
||||||
|
@ -205,7 +201,7 @@ void HMISong::SetupForHMI(int len)
|
||||||
}
|
}
|
||||||
|
|
||||||
// BTW, HMI does not actually check the track header.
|
// BTW, HMI does not actually check the track header.
|
||||||
if (memcmp(MusHeader + start, TRACK_MAGIC, 13) != 0)
|
if (memcmp(MusPtr + start, TRACK_MAGIC, 13) != 0)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -218,7 +214,7 @@ void HMISong::SetupForHMI(int len)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
tracklen = GetInt(MusHeader + track_dir + i*4 + 4) - start;
|
tracklen = GetInt(MusPtr + track_dir + i*4 + 4) - start;
|
||||||
}
|
}
|
||||||
// Clamp incomplete tracks to the end of the file.
|
// Clamp incomplete tracks to the end of the file.
|
||||||
tracklen = MIN(tracklen, len - start);
|
tracklen = MIN(tracklen, len - start);
|
||||||
|
@ -228,7 +224,7 @@ void HMISong::SetupForHMI(int len)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Offset to actual MIDI events.
|
// Offset to actual MIDI events.
|
||||||
datastart = GetInt(MusHeader + start + HMITRACK_DATA_PTR_OFFSET);
|
datastart = GetInt(MusPtr + start + HMITRACK_DATA_PTR_OFFSET);
|
||||||
tracklen -= datastart;
|
tracklen -= datastart;
|
||||||
if (tracklen <= 0)
|
if (tracklen <= 0)
|
||||||
{
|
{
|
||||||
|
@ -236,7 +232,7 @@ void HMISong::SetupForHMI(int len)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store track information
|
// Store track information
|
||||||
Tracks[p].TrackBegin = MusHeader + start + datastart;
|
Tracks[p].TrackBegin = MusPtr + start + datastart;
|
||||||
Tracks[p].TrackP = 0;
|
Tracks[p].TrackP = 0;
|
||||||
Tracks[p].MaxTrackP = tracklen;
|
Tracks[p].MaxTrackP = tracklen;
|
||||||
|
|
||||||
|
@ -244,7 +240,7 @@ void HMISong::SetupForHMI(int len)
|
||||||
// connected to the MIDI device.
|
// connected to the MIDI device.
|
||||||
for (int ii = 0; ii < NUM_HMI_DESIGNATIONS; ++ii)
|
for (int ii = 0; ii < NUM_HMI_DESIGNATIONS; ++ii)
|
||||||
{
|
{
|
||||||
Tracks[p].Designation[ii] = GetShort(MusHeader + start + HMITRACK_DESIGNATION_OFFSET + ii*2);
|
Tracks[p].Designation[ii] = GetShort(MusPtr + start + HMITRACK_DESIGNATION_OFFSET + ii*2);
|
||||||
}
|
}
|
||||||
|
|
||||||
p++;
|
p++;
|
||||||
|
@ -266,12 +262,14 @@ void HMISong::SetupForHMP(int len)
|
||||||
int track_data;
|
int track_data;
|
||||||
int i, p;
|
int i, p;
|
||||||
|
|
||||||
|
auto MusPtr = &MusHeader[0];
|
||||||
|
|
||||||
ReadVarLen = ReadVarLenHMP;
|
ReadVarLen = ReadVarLenHMP;
|
||||||
if (MusHeader[8] == 0)
|
if (MusPtr[8] == 0)
|
||||||
{
|
{
|
||||||
track_data = HMP_TRACK_OFFSET_0;
|
track_data = HMP_TRACK_OFFSET_0;
|
||||||
}
|
}
|
||||||
else if (memcmp(MusHeader + 8, HMP_NEW_DATE, sizeof(HMP_NEW_DATE)) == 0)
|
else if (memcmp(MusPtr + 8, HMP_NEW_DATE, sizeof(HMP_NEW_DATE)) == 0)
|
||||||
{
|
{
|
||||||
track_data = HMP_TRACK_OFFSET_1;
|
track_data = HMP_TRACK_OFFSET_1;
|
||||||
}
|
}
|
||||||
|
@ -280,7 +278,7 @@ void HMISong::SetupForHMP(int len)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
NumTracks = GetInt(MusHeader + HMP_TRACK_COUNT_OFFSET);
|
NumTracks = GetInt(MusPtr + HMP_TRACK_COUNT_OFFSET);
|
||||||
|
|
||||||
if (NumTracks <= 0)
|
if (NumTracks <= 0)
|
||||||
{
|
{
|
||||||
|
@ -288,7 +286,7 @@ void HMISong::SetupForHMP(int len)
|
||||||
}
|
}
|
||||||
|
|
||||||
// The division is the number of pulses per quarter note (PPQN).
|
// The division is the number of pulses per quarter note (PPQN).
|
||||||
Division = GetInt(MusHeader + HMP_DIVISION_OFFSET);
|
Division = GetInt(MusPtr + HMP_DIVISION_OFFSET);
|
||||||
Tempo = InitialTempo = 1000000;
|
Tempo = InitialTempo = 1000000;
|
||||||
|
|
||||||
Tracks = new TrackInfo[NumTracks + 1];
|
Tracks = new TrackInfo[NumTracks + 1];
|
||||||
|
@ -304,7 +302,7 @@ void HMISong::SetupForHMP(int len)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
tracklen = GetInt(MusHeader + start + HMPTRACK_LEN_OFFSET);
|
tracklen = GetInt(MusPtr + start + HMPTRACK_LEN_OFFSET);
|
||||||
track_data += tracklen;
|
track_data += tracklen;
|
||||||
|
|
||||||
// Clamp incomplete tracks to the end of the file.
|
// Clamp incomplete tracks to the end of the file.
|
||||||
|
@ -322,7 +320,7 @@ void HMISong::SetupForHMP(int len)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store track information
|
// Store track information
|
||||||
Tracks[p].TrackBegin = MusHeader + start + HMPTRACK_MIDI_DATA_OFFSET;
|
Tracks[p].TrackBegin = MusPtr + start + HMPTRACK_MIDI_DATA_OFFSET;
|
||||||
Tracks[p].TrackP = 0;
|
Tracks[p].TrackP = 0;
|
||||||
Tracks[p].MaxTrackP = tracklen;
|
Tracks[p].MaxTrackP = tracklen;
|
||||||
|
|
||||||
|
@ -333,14 +331,14 @@ void HMISong::SetupForHMP(int len)
|
||||||
// HMI files. Some songs contain nothing but zeroes for this data, so I'd rather
|
// HMI files. Some songs contain nothing but zeroes for this data, so I'd rather
|
||||||
// not go around using it without confirmation.
|
// not go around using it without confirmation.
|
||||||
|
|
||||||
Printf("Track %d: %d %08x %d: \034I", i, GetInt(MusHeader + start),
|
Printf("Track %d: %d %08x %d: \034I", i, GetInt(MusPtr + start),
|
||||||
GetInt(MusHeader + start + 4), GetInt(MusHeader + start + 8));
|
GetInt(MusPtr + start + 4), GetInt(MusPtr + start + 8));
|
||||||
|
|
||||||
int designations = HMP_DESIGNATIONS_OFFSET +
|
int designations = HMP_DESIGNATIONS_OFFSET +
|
||||||
GetInt(MusHeader + start + HMPTRACK_DESIGNATION_OFFSET) * 4 * NUM_HMP_DESIGNATIONS;
|
GetInt(MusPtr + start + HMPTRACK_DESIGNATION_OFFSET) * 4 * NUM_HMP_DESIGNATIONS;
|
||||||
for (int ii = 0; ii < NUM_HMP_DESIGNATIONS; ++ii)
|
for (int ii = 0; ii < NUM_HMP_DESIGNATIONS; ++ii)
|
||||||
{
|
{
|
||||||
Printf(" %04x", GetInt(MusHeader + designations + ii*4));
|
Printf(" %04x", GetInt(MusPtr + designations + ii*4));
|
||||||
}
|
}
|
||||||
Printf("\n");
|
Printf("\n");
|
||||||
#endif
|
#endif
|
||||||
|
@ -474,7 +472,7 @@ void HMISong :: DoRestart()
|
||||||
|
|
||||||
bool HMISong::CheckDone()
|
bool HMISong::CheckDone()
|
||||||
{
|
{
|
||||||
return TrackDue == NULL;
|
return TrackDue == nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -964,7 +962,7 @@ void NoteOffQueue::Heapify()
|
||||||
//
|
//
|
||||||
// HMISong :: FindNextDue
|
// HMISong :: FindNextDue
|
||||||
//
|
//
|
||||||
// Scans every track for the next event to play. Returns NULL if all events
|
// Scans every track for the next event to play. Returns nullptr if all events
|
||||||
// have been consumed.
|
// have been consumed.
|
||||||
//
|
//
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -987,7 +985,7 @@ HMISong::TrackInfo *HMISong::FindNextDue ()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check regular tracks.
|
// Check regular tracks.
|
||||||
track = NULL;
|
track = nullptr;
|
||||||
best = 0xFFFFFFFF;
|
best = 0xFFFFFFFF;
|
||||||
for (i = 0; i < NumTracks; ++i)
|
for (i = 0; i < NumTracks; ++i)
|
||||||
{
|
{
|
||||||
|
|
|
@ -98,13 +98,11 @@ struct MIDISong2::TrackInfo
|
||||||
MIDISong2::MIDISong2 (FileReader &reader)
|
MIDISong2::MIDISong2 (FileReader &reader)
|
||||||
: MusHeader(0), Tracks(0)
|
: MusHeader(0), Tracks(0)
|
||||||
{
|
{
|
||||||
int p;
|
unsigned p;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
SongLen = (int)reader.GetLength();
|
MusHeader = reader.Read();
|
||||||
MusHeader = new uint8_t[SongLen];
|
if (MusHeader.Size() == 0) return;
|
||||||
if (reader.Read(MusHeader, SongLen) != SongLen)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Do some validation of the MIDI file
|
// Do some validation of the MIDI file
|
||||||
if (MusHeader[4] != 0 || MusHeader[5] != 0 || MusHeader[6] != 0 || MusHeader[7] != 6)
|
if (MusHeader[4] != 0 || MusHeader[5] != 0 || MusHeader[6] != 0 || MusHeader[7] != 6)
|
||||||
|
@ -134,7 +132,7 @@ MIDISong2::MIDISong2 (FileReader &reader)
|
||||||
Tracks = new TrackInfo[NumTracks];
|
Tracks = new TrackInfo[NumTracks];
|
||||||
|
|
||||||
// Gather information about each track
|
// Gather information about each track
|
||||||
for (i = 0, p = 14; i < NumTracks && p < SongLen + 8; ++i)
|
for (i = 0, p = 14; i < NumTracks && p < MusHeader.Size() + 8; ++i)
|
||||||
{
|
{
|
||||||
uint32_t chunkLen =
|
uint32_t chunkLen =
|
||||||
(MusHeader[p+4]<<24) |
|
(MusHeader[p+4]<<24) |
|
||||||
|
@ -142,9 +140,9 @@ MIDISong2::MIDISong2 (FileReader &reader)
|
||||||
(MusHeader[p+6]<<8) |
|
(MusHeader[p+6]<<8) |
|
||||||
(MusHeader[p+7]);
|
(MusHeader[p+7]);
|
||||||
|
|
||||||
if (chunkLen + p + 8 > (uint32_t)SongLen)
|
if (chunkLen + p + 8 > MusHeader.Size())
|
||||||
{ // Track too long, so truncate it
|
{ // Track too long, so truncate it
|
||||||
chunkLen = SongLen - p - 8;
|
chunkLen = MusHeader.Size() - p - 8;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (MusHeader[p+0] == 'M' &&
|
if (MusHeader[p+0] == 'M' &&
|
||||||
|
@ -152,7 +150,7 @@ MIDISong2::MIDISong2 (FileReader &reader)
|
||||||
MusHeader[p+2] == 'r' &&
|
MusHeader[p+2] == 'r' &&
|
||||||
MusHeader[p+3] == 'k')
|
MusHeader[p+3] == 'k')
|
||||||
{
|
{
|
||||||
Tracks[i].TrackBegin = MusHeader + p + 8;
|
Tracks[i].TrackBegin = &MusHeader[p + 8];
|
||||||
Tracks[i].TrackP = 0;
|
Tracks[i].TrackP = 0;
|
||||||
Tracks[i].MaxTrackP = chunkLen;
|
Tracks[i].MaxTrackP = chunkLen;
|
||||||
}
|
}
|
||||||
|
@ -178,14 +176,10 @@ MIDISong2::MIDISong2 (FileReader &reader)
|
||||||
|
|
||||||
MIDISong2::~MIDISong2 ()
|
MIDISong2::~MIDISong2 ()
|
||||||
{
|
{
|
||||||
if (Tracks != NULL)
|
if (Tracks != nullptr)
|
||||||
{
|
{
|
||||||
delete[] Tracks;
|
delete[] Tracks;
|
||||||
}
|
}
|
||||||
if (MusHeader != NULL)
|
|
||||||
{
|
|
||||||
delete[] MusHeader;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -273,7 +267,7 @@ void MIDISong2 :: DoRestart()
|
||||||
|
|
||||||
bool MIDISong2::CheckDone()
|
bool MIDISong2::CheckDone()
|
||||||
{
|
{
|
||||||
return TrackDue == NULL;
|
return TrackDue == nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -750,7 +744,7 @@ uint32_t MIDISong2::TrackInfo::ReadVarLen ()
|
||||||
//
|
//
|
||||||
// MIDISong2 :: FindNextDue
|
// MIDISong2 :: FindNextDue
|
||||||
//
|
//
|
||||||
// Scans every track for the next event to play. Returns NULL if all events
|
// Scans every track for the next event to play. Returns nullptr if all events
|
||||||
// have been consumed.
|
// have been consumed.
|
||||||
//
|
//
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -770,10 +764,10 @@ MIDISong2::TrackInfo *MIDISong2::FindNextDue ()
|
||||||
switch (Format)
|
switch (Format)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
return Tracks[0].Finished ? NULL : Tracks;
|
return Tracks[0].Finished ? nullptr : Tracks;
|
||||||
|
|
||||||
case 1:
|
case 1:
|
||||||
track = NULL;
|
track = nullptr;
|
||||||
best = 0xFFFFFFFF;
|
best = 0xFFFFFFFF;
|
||||||
for (i = 0; i < NumTracks; ++i)
|
for (i = 0; i < NumTracks; ++i)
|
||||||
{
|
{
|
||||||
|
@ -794,9 +788,9 @@ MIDISong2::TrackInfo *MIDISong2::FindNextDue ()
|
||||||
{
|
{
|
||||||
track++;
|
track++;
|
||||||
}
|
}
|
||||||
return track < &Tracks[NumTracks] ? track : NULL;
|
return track < &Tracks[NumTracks] ? track : nullptr;
|
||||||
}
|
}
|
||||||
return NULL;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -104,13 +104,11 @@ struct XMISong::TrackInfo
|
||||||
XMISong::XMISong (FileReader &reader)
|
XMISong::XMISong (FileReader &reader)
|
||||||
: MusHeader(0), Songs(0)
|
: MusHeader(0), Songs(0)
|
||||||
{
|
{
|
||||||
SongLen = (int)reader.GetLength();
|
MusHeader = reader.Read();
|
||||||
MusHeader = new uint8_t[SongLen];
|
if (MusHeader.Size() == 0) return;
|
||||||
if (reader.Read(MusHeader, SongLen) != SongLen)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Find all the songs in this file.
|
// Find all the songs in this file.
|
||||||
NumSongs = FindXMIDforms(MusHeader, SongLen, NULL);
|
NumSongs = FindXMIDforms(&MusHeader[0], MusHeader.Size(), nullptr);
|
||||||
if (NumSongs == 0)
|
if (NumSongs == 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
|
@ -128,7 +126,7 @@ XMISong::XMISong (FileReader &reader)
|
||||||
|
|
||||||
Songs = new TrackInfo[NumSongs];
|
Songs = new TrackInfo[NumSongs];
|
||||||
memset(Songs, 0, sizeof(*Songs) * NumSongs);
|
memset(Songs, 0, sizeof(*Songs) * NumSongs);
|
||||||
FindXMIDforms(MusHeader, SongLen, Songs);
|
FindXMIDforms(&MusHeader[0], MusHeader.Size(), Songs);
|
||||||
CurrSong = Songs;
|
CurrSong = Songs;
|
||||||
DPrintf(DMSG_SPAMMY, "XMI song count: %d\n", NumSongs);
|
DPrintf(DMSG_SPAMMY, "XMI song count: %d\n", NumSongs);
|
||||||
}
|
}
|
||||||
|
@ -141,14 +139,10 @@ XMISong::XMISong (FileReader &reader)
|
||||||
|
|
||||||
XMISong::~XMISong ()
|
XMISong::~XMISong ()
|
||||||
{
|
{
|
||||||
if (Songs != NULL)
|
if (Songs != nullptr)
|
||||||
{
|
{
|
||||||
delete[] Songs;
|
delete[] Songs;
|
||||||
}
|
}
|
||||||
if (MusHeader != NULL)
|
|
||||||
{
|
|
||||||
delete[] MusHeader;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -172,7 +166,7 @@ int XMISong::FindXMIDforms(const uint8_t *chunk, int len, TrackInfo *songs) cons
|
||||||
{
|
{
|
||||||
if (GetNativeInt(chunk + p + 8) == MAKE_ID('X','M','I','D'))
|
if (GetNativeInt(chunk + p + 8) == MAKE_ID('X','M','I','D'))
|
||||||
{
|
{
|
||||||
if (songs != NULL)
|
if (songs != nullptr)
|
||||||
{
|
{
|
||||||
FoundXMID(chunk + p + 12, chunklen - 4, songs + count);
|
FoundXMID(chunk + p + 12, chunklen - 4, songs + count);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue