- 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:
Christoph Oelckers 2018-11-02 10:20:12 +01:00
parent c07aeb7498
commit 1ccbbcb81d
5 changed files with 62 additions and 71 deletions

View file

@ -184,6 +184,14 @@ public:
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)
{
return mReader->Gets(strbuf, (int)len);

View file

@ -121,8 +121,7 @@ private:
uint32_t *SendCommand (uint32_t *event, TrackInfo *track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
TrackInfo *FindNextDue ();
uint8_t *MusHeader;
int SongLen;
TArray<uint8_t> MusHeader;
TrackInfo *Tracks;
TrackInfo *TrackDue;
int NumTracks;
@ -181,8 +180,7 @@ private:
static uint32_t ReadVarLenHMI(TrackInfo *);
static uint32_t ReadVarLenHMP(TrackInfo *);
uint8_t *MusHeader;
int SongLen;
TArray<uint8_t> MusHeader;
int NumTracks;
TrackInfo *Tracks;
TrackInfo *TrackDue;
@ -218,8 +216,7 @@ private:
uint32_t *SendCommand (uint32_t *event, EventSource track, uint32_t delay, ptrdiff_t room, bool &sysex_noroom);
EventSource FindNextDue();
uint8_t *MusHeader;
int SongLen; // length of the entire file
TArray<uint8_t> MusHeader;
int NumSongs;
TrackInfo *Songs;
TrackInfo *CurrSong;

View file

@ -129,19 +129,17 @@ HMISong::HMISong (FileReader &reader)
{ // Way too small to be HMI.
return;
}
MusHeader = new uint8_t[len];
SongLen = len;
MusHeader = reader.Read();
NumTracks = 0;
if (reader.Read(MusHeader, len) != len)
if (MusHeader.Size() == 0)
return;
// 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);
}
else if (((uint32_t *)MusHeader)[0] == MAKE_ID('H','M','I','M') &&
((uint32_t *)MusHeader)[1] == MAKE_ID('I','D','I','P'))
else if (memcmp(&MusHeader[0], "HMIMIDIP", 8) == 0)
{
SetupForHMP(len);
}
@ -155,14 +153,10 @@ HMISong::HMISong (FileReader &reader)
HMISong::~HMISong()
{
if (Tracks != NULL)
if (Tracks != nullptr)
{
delete[] Tracks;
}
if (MusHeader != NULL)
{
delete[] MusHeader;
}
}
//==========================================================================
@ -175,8 +169,10 @@ void HMISong::SetupForHMI(int len)
{
int i, p;
auto MusPtr = &MusHeader[0];
ReadVarLen = ReadVarLenHMI;
NumTracks = GetShort(MusHeader + HMI_TRACK_COUNT_OFFSET);
NumTracks = GetShort(MusPtr + HMI_TRACK_COUNT_OFFSET);
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,
// 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.
Division = GetShort(MusHeader + HMI_DIVISION_OFFSET) << 2;
Division = GetShort(MusPtr + HMI_DIVISION_OFFSET) << 2;
Tempo = InitialTempo = 4000000;
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
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;
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.
if (memcmp(MusHeader + start, TRACK_MAGIC, 13) != 0)
if (memcmp(MusPtr + start, TRACK_MAGIC, 13) != 0)
{
continue;
}
@ -218,7 +214,7 @@ void HMISong::SetupForHMI(int len)
}
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.
tracklen = MIN(tracklen, len - start);
@ -228,7 +224,7 @@ void HMISong::SetupForHMI(int len)
}
// Offset to actual MIDI events.
datastart = GetInt(MusHeader + start + HMITRACK_DATA_PTR_OFFSET);
datastart = GetInt(MusPtr + start + HMITRACK_DATA_PTR_OFFSET);
tracklen -= datastart;
if (tracklen <= 0)
{
@ -236,7 +232,7 @@ void HMISong::SetupForHMI(int len)
}
// Store track information
Tracks[p].TrackBegin = MusHeader + start + datastart;
Tracks[p].TrackBegin = MusPtr + start + datastart;
Tracks[p].TrackP = 0;
Tracks[p].MaxTrackP = tracklen;
@ -244,7 +240,7 @@ void HMISong::SetupForHMI(int len)
// connected to the MIDI device.
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++;
@ -266,12 +262,14 @@ void HMISong::SetupForHMP(int len)
int track_data;
int i, p;
auto MusPtr = &MusHeader[0];
ReadVarLen = ReadVarLenHMP;
if (MusHeader[8] == 0)
if (MusPtr[8] == 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;
}
@ -280,7 +278,7 @@ void HMISong::SetupForHMP(int len)
return;
}
NumTracks = GetInt(MusHeader + HMP_TRACK_COUNT_OFFSET);
NumTracks = GetInt(MusPtr + HMP_TRACK_COUNT_OFFSET);
if (NumTracks <= 0)
{
@ -288,7 +286,7 @@ void HMISong::SetupForHMP(int len)
}
// 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;
Tracks = new TrackInfo[NumTracks + 1];
@ -304,7 +302,7 @@ void HMISong::SetupForHMP(int len)
break;
}
tracklen = GetInt(MusHeader + start + HMPTRACK_LEN_OFFSET);
tracklen = GetInt(MusPtr + start + HMPTRACK_LEN_OFFSET);
track_data += tracklen;
// Clamp incomplete tracks to the end of the file.
@ -322,7 +320,7 @@ void HMISong::SetupForHMP(int len)
}
// 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].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
// not go around using it without confirmation.
Printf("Track %d: %d %08x %d: \034I", i, GetInt(MusHeader + start),
GetInt(MusHeader + start + 4), GetInt(MusHeader + start + 8));
Printf("Track %d: %d %08x %d: \034I", i, GetInt(MusPtr + start),
GetInt(MusPtr + start + 4), GetInt(MusPtr + start + 8));
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)
{
Printf(" %04x", GetInt(MusHeader + designations + ii*4));
Printf(" %04x", GetInt(MusPtr + designations + ii*4));
}
Printf("\n");
#endif
@ -474,7 +472,7 @@ void HMISong :: DoRestart()
bool HMISong::CheckDone()
{
return TrackDue == NULL;
return TrackDue == nullptr;
}
//==========================================================================
@ -964,7 +962,7 @@ void NoteOffQueue::Heapify()
//
// 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.
//
//==========================================================================
@ -987,7 +985,7 @@ HMISong::TrackInfo *HMISong::FindNextDue ()
}
// Check regular tracks.
track = NULL;
track = nullptr;
best = 0xFFFFFFFF;
for (i = 0; i < NumTracks; ++i)
{

View file

@ -98,13 +98,11 @@ struct MIDISong2::TrackInfo
MIDISong2::MIDISong2 (FileReader &reader)
: MusHeader(0), Tracks(0)
{
int p;
unsigned p;
int i;
SongLen = (int)reader.GetLength();
MusHeader = new uint8_t[SongLen];
if (reader.Read(MusHeader, SongLen) != SongLen)
return;
MusHeader = reader.Read();
if (MusHeader.Size() == 0) return;
// Do some validation of the MIDI file
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];
// 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 =
(MusHeader[p+4]<<24) |
@ -142,9 +140,9 @@ MIDISong2::MIDISong2 (FileReader &reader)
(MusHeader[p+6]<<8) |
(MusHeader[p+7]);
if (chunkLen + p + 8 > (uint32_t)SongLen)
if (chunkLen + p + 8 > MusHeader.Size())
{ // Track too long, so truncate it
chunkLen = SongLen - p - 8;
chunkLen = MusHeader.Size() - p - 8;
}
if (MusHeader[p+0] == 'M' &&
@ -152,7 +150,7 @@ MIDISong2::MIDISong2 (FileReader &reader)
MusHeader[p+2] == 'r' &&
MusHeader[p+3] == 'k')
{
Tracks[i].TrackBegin = MusHeader + p + 8;
Tracks[i].TrackBegin = &MusHeader[p + 8];
Tracks[i].TrackP = 0;
Tracks[i].MaxTrackP = chunkLen;
}
@ -178,14 +176,10 @@ MIDISong2::MIDISong2 (FileReader &reader)
MIDISong2::~MIDISong2 ()
{
if (Tracks != NULL)
if (Tracks != nullptr)
{
delete[] Tracks;
}
if (MusHeader != NULL)
{
delete[] MusHeader;
}
}
//==========================================================================
@ -273,7 +267,7 @@ void MIDISong2 :: DoRestart()
bool MIDISong2::CheckDone()
{
return TrackDue == NULL;
return TrackDue == nullptr;
}
//==========================================================================
@ -750,7 +744,7 @@ uint32_t MIDISong2::TrackInfo::ReadVarLen ()
//
// 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.
//
//==========================================================================
@ -770,10 +764,10 @@ MIDISong2::TrackInfo *MIDISong2::FindNextDue ()
switch (Format)
{
case 0:
return Tracks[0].Finished ? NULL : Tracks;
return Tracks[0].Finished ? nullptr : Tracks;
case 1:
track = NULL;
track = nullptr;
best = 0xFFFFFFFF;
for (i = 0; i < NumTracks; ++i)
{
@ -794,9 +788,9 @@ MIDISong2::TrackInfo *MIDISong2::FindNextDue ()
{
track++;
}
return track < &Tracks[NumTracks] ? track : NULL;
return track < &Tracks[NumTracks] ? track : nullptr;
}
return NULL;
return nullptr;
}

View file

@ -104,13 +104,11 @@ struct XMISong::TrackInfo
XMISong::XMISong (FileReader &reader)
: MusHeader(0), Songs(0)
{
SongLen = (int)reader.GetLength();
MusHeader = new uint8_t[SongLen];
if (reader.Read(MusHeader, SongLen) != SongLen)
return;
MusHeader = reader.Read();
if (MusHeader.Size() == 0) return;
// Find all the songs in this file.
NumSongs = FindXMIDforms(MusHeader, SongLen, NULL);
NumSongs = FindXMIDforms(&MusHeader[0], MusHeader.Size(), nullptr);
if (NumSongs == 0)
{
return;
@ -128,7 +126,7 @@ XMISong::XMISong (FileReader &reader)
Songs = new TrackInfo[NumSongs];
memset(Songs, 0, sizeof(*Songs) * NumSongs);
FindXMIDforms(MusHeader, SongLen, Songs);
FindXMIDforms(&MusHeader[0], MusHeader.Size(), Songs);
CurrSong = Songs;
DPrintf(DMSG_SPAMMY, "XMI song count: %d\n", NumSongs);
}
@ -141,14 +139,10 @@ XMISong::XMISong (FileReader &reader)
XMISong::~XMISong ()
{
if (Songs != NULL)
if (Songs != nullptr)
{
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 (songs != NULL)
if (songs != nullptr)
{
FoundXMID(chunk + p + 12, chunklen - 4, songs + count);
}