bool MIDIFileReadMultiTrack::mf_sysex ( MIDIClockTime time, int type, int len, unsigned char *s ) // funcVRM
{
    MIDITimedMessage msg;
    msg.SetSysEx( type ); // set msg status byte (0xF0 or 0xF7)

    int num = len; // number of possible SysExURT header data bytes in msg, 0...5
    if ( num > 5 )
        num = 5;

    // add up to 5 starting bytes for SysExURT functions
    if ( num > 0 )
        msg.SetByte2( s[0] );
    if ( num > 1 )
        msg.SetByte3( s[1] );
    if ( num > 2 )
        msg.SetByte4( s[2] );
    if ( num > 3 )
        msg.SetByte5( s[3] );
    if ( num > 4 )
        msg.SetByte6( s[4] );

    msg.SetTime ( time );
    MIDISystemExclusive sysex( len );

    for ( int i = 0; i < len; ++i )
    {
        sysex.PutSysByte ( s[i] );
    }

   msg.SetDataLength( num );
   return AddEventToMultiTrack ( msg, &sysex, cur_track );
}
bool MIDIFileReadMultiTrack::mf_metamisc ( MIDIClockTime time, int type, int len, unsigned char *data ) // funcVRM
{
    // code for all miscellaneous meta events

    MIDITimedMessage msg;
    msg.SetTime ( time );

    msg.SetStatus( META_EVENT );
    msg.SetByte1( type );

    if ( len <= 5 )
    {
        if ( len > 0 )
            msg.SetByte2( data[0] );

        if ( len > 1 )
            msg.SetByte3( data[1] );

        if ( len > 2 )
            msg.SetByte4( data[2] );

        if ( len > 3 )
            msg.SetByte5( data[3] );

        if ( len > 4 )
            msg.SetByte6( data[4] );
    }
    // else msg add to track, but do'nt write to output midifile!

    msg.SetDataLength( len );
    return AddEventToMultiTrack ( msg, 0, cur_track );
}
bool MIDIFileReadMultiTrack::mf_keysig ( MIDIClockTime time, int c, int v ) // VRM
{
    MIDITimedMessage msg;
    msg.SetTime ( time );
    msg.SetKeySig ( ( unsigned char ) c, ( unsigned char ) v );
    msg.SetDataLength( 2 ); // VRM
    return AddEventToMultiTrack ( msg, 0, cur_track ); // VRM
}
bool MIDIFileReadMultiTrack::mf_timesig ( MIDIClockTime time, int num, int den_pow, int clks_per_metro, int notated_32nd_per_quarter ) // funcVRM
{
    MIDITimedMessage msg;
    msg.SetTime ( time );
    msg.SetTimeSig ( num, den_pow, clks_per_metro, notated_32nd_per_quarter );
    msg.SetDataLength( 5 ); // VRM // source 4 bytes + 1 byte for denominator
    return AddEventToMultiTrack ( msg, 0, cur_track );
}
bool MIDIFileReadMultiTrack::mf_eot ( MIDIClockTime time ) // VRM
{
    MIDITimedMessage msg;
    msg.SetStatus ( META_EVENT );
    msg.SetMetaType ( META_END_OF_TRACK ); // VRM
    msg.SetTime ( time );
    msg.SetDataLength( 0 ); // VRM
    return AddEventToMultiTrack ( msg, 0, cur_track ); // VRM
}
bool MIDIFileReadMultiTrack::mf_tempo ( MIDIClockTime time, unsigned char a, unsigned char b, unsigned char c ) // funcVRM
{
    MIDITimedMessage msg;
    msg.SetTime ( time );
    msg.SetMetaEvent ( MF_META_TEMPO, a, b );
    msg.SetByte4( c );
    msg.SetDataLength( 3 );
    return AddEventToMultiTrack ( msg, 0, cur_track );
}
bool MIDIFileReadMultiTrack::mf_text ( MIDIClockTime time, int type, int len, unsigned char *s ) // VRM
{
    MIDITimedMessage msg;
    msg.SetStatus ( META_EVENT );
    msg.SetMetaType ( ( uchar ) type ); // remember - MF_META_* id codes match META_* codes
    msg.SetTime ( time );

    MIDISystemExclusive sysex( len ); // VRM

    for ( int i = 0; i < len; ++i )
    {
        sysex.PutSysByte ( s[i] ); // VRM
    }

    msg.SetDataLength( 0 ); // VRM // variable data length don't saved to data_length
    return AddEventToMultiTrack ( msg, &sysex, cur_track ); // VRM
}