#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <stdint.h>
#include <string.h>
//midi event types
enum {
NOTE_OFF = 0x8 ,
NOTE_ON = 0x9 ,
NOTE_AFTERTOUCH = 0xA ,
CONTROLLER = 0xB ,
PROGRAM_CHANGE = 0xC ,
PROGRAM_AFTERTOUCH = 0xD ,
PITCH_BEND = 0xE ,
} ;
//meta event types
enum {
SEQUENCE_NUMBER = 0x00 ,
TEXT = 0x01 ,
COPYRIGHT_NOTICE = 0x02 ,
SEQUENCE_TRACK_NAME = 0x03 ,
INSTRUMENT_NAME = 0x04 ,
LYRICS = 0x05 ,
MARKER = 0x06 ,
CUE_POINT = 0x07 ,
//UNKNOWN_TEXT = 0x09;
MIDI_CHANNEL_PREFIX = 0x20 ,
END_OF_TRACK = 0x2F ,
SET_TEMPO = 0x51 ,
SMPTE_OFFSET = 0x54 ,
TIME_SIGNATURE = 0x58 ,
KEY_SIGNATURE = 0x59 ,
SEQUENCE_SPECIFIC = 0x7F ,
} ;
//system exclusive events
/*enum{
NORMAL = 0xF0;
DIVIDED = 0xF0;
DIVIDED = 0xF7;
};*/
typedef struct _MthdCD{
uint16_t format;
uint16_t tracks_count;
uint16_t time_division;
uint8_t time_divisionType;
} _MthdCD;
typedef struct _metaEventStruct{
uint8_t type;
uint64_t length;
uint8_t * data;
} _metaEventStruct;
typedef struct _midiEventStruct{
uint8_t type;
uint8_t channel;
uint8_t param1, param2;
} _midiEventStruct;
typedef struct _sysExEventStruct{
uint64_t length;
uint8_t * data;
} _sysExEventStruct;
typedef struct _MtrkCD{
uint32_t byteCount;
uint64_t elementCount;
uint64_t * deltaTime;
uint8_t * eventType;
_metaEventStruct * metaEventValues;
_midiEventStruct * midiEventValues;
_sysExEventStruct * sysExEventValues;
} _MtrkCD;
uint64_t _ReadVariableLength( FILE * file, int * l) ;
uint8_t _ReadByte( FILE * file) ;
uint16_t _ReadByte2( FILE * file) ;
uint32_t _ReadByte4( FILE * file) ;
uint8_t _firstBits4( int byte) ;
uint8_t _lastBits4( int byte) ;
uint8_t _firstByte( int bytes) ;
uint8_t _lastByte( int bytes) ;
int _ReadMthdChunk( FILE * file, _MthdCD * MthdChunkData, int print) ;
int _ReadMtrkChunk( FILE * file, _MtrkCD * MtrkChunkData, int trackID, int print) ;
int _ParseMidi( char * fileName, int print, int save) ;
uint64_t _ReadVariableLength( FILE * file, int * l) {
* l= 0 ;
uint64_t totalVal = 0 ;
uint8_t curVal= 0 , copyVal= 0 ;
do {
copyVal = curVal = _ReadByte( file) ;
copyVal &= ~( 1 << 7 ) ;
totalVal = ( totalVal << 7 ) + copyVal;
( * l) ++;
} while ( curVal& ( 1 << 7 ) ) ;
return totalVal;
}
uint8_t _ReadByte( FILE * file) {
}
uint16_t _ReadByte2( FILE * file) {
uint16_t res
= fgetc ( file
) ; res
= ( res
<< 8 ) + fgetc ( file
) ; return res;
}
uint32_t _ReadByte4( FILE * file) {
uint32_t res
= fgetc ( file
) ; res
= ( res
<< 8 ) + fgetc ( file
) ; res
= ( res
<< 8 ) + fgetc ( file
) ; res
= ( res
<< 8 ) + fgetc ( file
) ; return res;
}
uint8_t _firstBits4( int byte) {
return ( byte>> 4 ) ;
}
uint8_t _lastBits4( int byte) {
return byte- ( ( byte>> 4 ) << 4 ) ;
}
uint8_t _firstByte( int bytes) {
return ( bytes>> 8 ) ;
}
uint8_t _lastByte( int bytes) {
return bytes- ( ( bytes>> 8 ) << 8 ) ;
}
int _ReadMthdChunk( FILE * file, _MthdCD * MthdChunkData, int print) {
uint32_t chkName = _ReadByte4( file) ;
if ( chkName != 0x4D546864 ) {
return 0 ;
}
uint32_t chkLength = _ReadByte4( file) ;
if ( chkLength != 0x00000006 ) {
return 0 ;
}
MthdChunkData-> format = _ReadByte2( file) ;
MthdChunkData-> tracks_count = _ReadByte2( file) ;
MthdChunkData-> time_division = _ReadByte2( file) ;
if ( MthdChunkData-> time_divisionType & ( 1 << 16 ) ) {
MthdChunkData-> time_divisionType = 1 ;
}
else {
MthdChunkData-> time_divisionType = 0 ;
}
if ( print) {
printf ( "\n \n \n --------------------------------------------------------------------------------\n Parsed MThd:\n \n " ) ; printf ( "Chunk Name: %26s%c%c%c%c\n " , "" , ( char ) ( chkName
>> 24 ) , ( char ) ( ( chkName
>> 16 ) % 0x100 ) , ( char ) ( ( chkName
>> 8 ) % 0x100 ) , ( char ) ( ( chkName
) % 0x100 ) ) ; printf ( "Chunk Size: %24d bytes\n \n " , chkLength
) ; char str[ 25 ] ;
switch ( MthdChunkData-> format) {
case 0x00000000 :
break ;
case 0x00000001 :
break ;
case 0x00000002 :
break ;
default :
return 0 ;
}
printf ( "Midi Format: %29s\n " , str
) ; printf ( "Number of Tracks: %24d\n " , MthdChunkData
-> tracks_count
) ; switch ( MthdChunkData-> time_divisionType) {
case 0x00 :
strcpy ( str
, "Ticks per Quarter Note" ) ; break ;
case 0x01 :
strcpy ( str
, "Frames per Second" ) ; break ;
default :
return 0 ;
}
printf ( "Time Division Type: %22s\n Time Division Value: %21d\n " , str
, MthdChunkData
-> time_division
) ; }
return 1 ;
}
int _ReadMtrkChunk( FILE * file, _MtrkCD * MtrkChunkData, int trackID, int print) {
//static int trackID = 0;
int i, j;
uint32_t chkName = _ReadByte4( file) ;
if ( chkName != 0x4D54726B ) {
return 0 ;
}
MtrkChunkData-> byteCount = _ReadByte4( file) ;
uint32_t parsedLength = MtrkChunkData-> byteCount;
uint8_t midiEventType;
uint8_t midiChannel;
int l= 0 ;
MtrkChunkData-> elementCount= 0 ;
MtrkChunkData
-> deltaTime
= ( uint64_t * ) ( malloc ( sizeof ( uint64_t ) ) ) ; MtrkChunkData
-> eventType
= ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) ) ) ; MtrkChunkData
-> metaEventValues
= ( _metaEventStruct
* ) ( malloc ( sizeof ( _metaEventStruct
) ) ) ; MtrkChunkData
-> midiEventValues
= ( _midiEventStruct
* ) ( malloc ( sizeof ( _midiEventStruct
) ) ) ; MtrkChunkData
-> sysExEventValues
= ( _sysExEventStruct
* ) ( malloc ( sizeof ( _sysExEventStruct
) ) ) ;
while ( parsedLength > 0 ) {
MtrkChunkData
-> deltaTime
= ( realloc ( MtrkChunkData
-> deltaTime
, sizeof ( uint64_t ) * ( MtrkChunkData
-> elementCount
+ 1 ) ) ) ; MtrkChunkData
-> eventType
= ( realloc ( MtrkChunkData
-> eventType
, sizeof ( uint8_t ) * ( MtrkChunkData
-> elementCount
+ 1 ) ) ) ; MtrkChunkData
-> metaEventValues
= ( realloc ( MtrkChunkData
-> metaEventValues
, sizeof ( _metaEventStruct
) * ( MtrkChunkData
-> elementCount
+ 1 ) ) ) ; MtrkChunkData
-> midiEventValues
= ( realloc ( MtrkChunkData
-> midiEventValues
, sizeof ( _midiEventStruct
) * ( MtrkChunkData
-> elementCount
+ 1 ) ) ) ; MtrkChunkData
-> sysExEventValues
= ( realloc ( MtrkChunkData
-> sysExEventValues
, sizeof ( _sysExEventStruct
) * ( MtrkChunkData
-> elementCount
+ 1 ) ) ) ;
//getch();
//printf("looped... toBeParsedLength is %d \n",parsedLength);
( ( MtrkChunkData-> deltaTime) [ MtrkChunkData-> elementCount] ) = _ReadVariableLength( file,& l) ;
//printf("Variable length: %d\n",MtrkChunkData->deltaTime);
parsedLength-= l;
uint8_t nByte = _ReadByte( file) ;
//printf("nByte: 0x%X\n",nByte);
parsedLength-= 1 ;
//check nByte for event type
switch ( nByte) {
//meta event case
case 0xFF :
( ( MtrkChunkData-> eventType) [ MtrkChunkData-> elementCount] ) = 1 ;
//printf("Found META event\n");
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .type = _ReadByte( file) ;
//printf("Meta event 0x%X\n",metaEventType);
parsedLength-= 1 ;
switch ( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .type ) {
//might seem trivial to seperate types since all is scanned the same way but it is actually needed for separating conversion and track combination
case SEQUENCE_NUMBER:
case KEY_SIGNATURE:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadByte( file) ;
parsedLength-= 1 ;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case SEQUENCE_SPECIFIC:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case TEXT:
case COPYRIGHT_NOTICE:
case SEQUENCE_TRACK_NAME:
case INSTRUMENT_NAME:
case LYRICS:
case MARKER:
case CUE_POINT:
case 0x09 : //unknown text event
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = '\0 ' ;
break ;
case MIDI_CHANNEL_PREFIX:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case SET_TEMPO:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case SMPTE_OFFSET:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case TIME_SIGNATURE:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
case END_OF_TRACK:
( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> metaEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> metaEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
default :
return 0 ;
}
break ;
//system exclusive event case
case 0xF0 :
case 0xF7 :
( ( MtrkChunkData-> eventType) [ MtrkChunkData-> elementCount] ) = 2 ;
//printf("Found System Exclusive event\n");
( ( MtrkChunkData-> sysExEventValues) [ MtrkChunkData-> elementCount] ) .length = _ReadVariableLength( file,& l) ;
parsedLength-= l;
( ( MtrkChunkData
-> sysExEventValues
) [ MtrkChunkData
-> elementCount
] ) .
data = ( uint8_t * ) ( malloc ( sizeof ( uint8_t ) * ( ( MtrkChunkData
-> sysExEventValues
) [ MtrkChunkData
-> elementCount
] ) .
length ) ) ; for ( i= 0 ; i< ( ( MtrkChunkData-> sysExEventValues) [ MtrkChunkData-> elementCount] ) .length ; i++ ) {
( ( ( MtrkChunkData-> sysExEventValues) [ MtrkChunkData-> elementCount] ) .data ) [ i] = _ReadByte( file) ;
parsedLength-= 1 ;
}
break ;
//midi event case
default :
( ( MtrkChunkData-> eventType) [ MtrkChunkData-> elementCount] ) = 0 ;
//printf("Found MIDI event\n");
midiEventType = _firstBits4( nByte) ;
midiChannel = _lastBits4( nByte) ;
//printf("MIDI event type: 0x%X\n",midiEventType);
switch ( midiEventType) {
case NOTE_OFF:
case NOTE_ON:
case NOTE_AFTERTOUCH:
case CONTROLLER:
case PITCH_BEND:
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .type = midiEventType;
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .channel = midiChannel;
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .param1 = _ReadByte( file) ;
parsedLength-= 1 ;
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] .param2 ) = _ReadByte( file) ;
parsedLength-= 1 ;
break ;
case PROGRAM_CHANGE:
case PROGRAM_AFTERTOUCH:
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .type = midiEventType;
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .channel = midiChannel;
( ( MtrkChunkData-> midiEventValues) [ MtrkChunkData-> elementCount] ) .param1 = _ReadByte( file) ;
parsedLength-= 1 ;
break ;
default :
return 0 ;
}
break ;
}
( MtrkChunkData-> elementCount) ++;
}
if ( print) {
printf ( "\n \n \n --------------------------------------------------------------------------------\n " ) ; printf ( "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n " ) ; printf ( "--------------------------------------------------------------------------------\n Parsed MTrk[%d]:\n \n " , trackID
) ; printf ( "Chunk Name: %26s%c%c%c%c\n " , "" , ( char ) ( chkName
>> 24 ) , ( char ) ( ( chkName
>> 16 ) % 0x100 ) , ( char ) ( ( chkName
>> 8 ) % 0x100 ) , ( char ) ( ( chkName
) % 0x100 ) ) ; printf ( "Chunk Size: %24d bytes\n " , MtrkChunkData
-> byteCount
) ; printf ( "Number of Events: %24d\n " , MtrkChunkData
-> elementCount
) ; char str[ 40 ] = { '\0 ' } ;
for ( i= 0 ; i< MtrkChunkData-> elementCount; i++ ) {
printf ( "\n Delta Time Value: %18d ticks\n " , MtrkChunkData
-> deltaTime
[ i
] ) ; switch ( MtrkChunkData-> eventType[ i] ) {
case 2 :
strcpy ( str
, "System Exclusive Event" ) ; break ;
case 1 :
break ;
case 0 :
break ;
}
printf ( "Event Type: %30s\n " , str
) ; if ( MtrkChunkData-> eventType[ i] == 1 ) {
switch ( MtrkChunkData-> metaEventValues[ i] .type ) {
case SEQUENCE_NUMBER:
strcpy ( str
, "Sequence Number Event" ) ; break ;
case TEXT:
break ;
case COPYRIGHT_NOTICE:
strcpy ( str
, "Copyright Notice Event" ) ; break ;
case SEQUENCE_TRACK_NAME:
strcpy ( str
, "Sequence / Track Name Event" ) ; break ;
case INSTRUMENT_NAME:
strcpy ( str
, "Instrument Name Event" ) ; break ;
case LYRICS:
break ;
case MARKER:
break ;
case CUE_POINT:
strcpy ( str
, "Cue Point Event" ) ; break ;
case 0x09 :
strcpy ( str
, "Unknown Text Type Event" ) ; break ;
case MIDI_CHANNEL_PREFIX:
strcpy ( str
, "MIDI Channel Prefix Event" ) ; break ;
case END_OF_TRACK:
strcpy ( str
, "End of Track Event" ) ; break ;
case SET_TEMPO:
strcpy ( str
, "Set Tempo Event" ) ; break ;
case SMPTE_OFFSET:
strcpy ( str
, "SMPTE Offset Event" ) ; break ;
case TIME_SIGNATURE:
strcpy ( str
, "Time Signature Event" ) ; break ;
case KEY_SIGNATURE:
strcpy ( str
, "Key Signature Event" ) ; break ;
case SEQUENCE_SPECIFIC:
strcpy ( str
, "Sequencer Specific Event" ) ; break ;
default :
}
printf ( "Event Sub-Type: %26s\n " , str
) ; switch ( MtrkChunkData-> metaEventValues[ i] .type ) {
case SEQUENCE_NUMBER:
printf ( "Event Parameters:\n " ) ; printf ( "Value: %0x04X\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] << 8 + MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] ) ; break ;
case TEXT:
case COPYRIGHT_NOTICE:
case SEQUENCE_TRACK_NAME:
case INSTRUMENT_NAME:
case LYRICS:
case MARKER:
case CUE_POINT:
case 0x09 :
printf ( "Event Parameters:\n " ) ; printf ( "%s\n " , ( char * ) MtrkChunkData
-> metaEventValues
[ i
] .
data ) ; break ;
case MIDI_CHANNEL_PREFIX:
printf ( "Event Parameters:\n " ) ; printf ( "Channel: %d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] ) ; break ;
case SET_TEMPO:
printf ( "Event Parameters:\n " ) ; printf ( "Value: %8u Microseconds per Quarter Note\n " , ( MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] << 16 ) + ( MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] << 8 ) + MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] ) ; break ;
case SMPTE_OFFSET:
printf ( "Event Parameters:\n " ) ; printf ( "Hour: %-2d Min: %-2d Sec: %-2d Frames: %-2d Sub-Frames: %-2d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 3 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 4 ] ) ; break ;
case TIME_SIGNATURE:
printf ( "Event Parameters:\n " ) ; printf ( "Numerator: %-3d Denominator: %-3d Metronome: %-3d 1/32: %-3d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 3 ] ) ; break ;
case KEY_SIGNATURE:
printf ( "Event Parameters:\n " ) ; printf ( "Key: %+2d Scale: %d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] ) ; break ;
default :
break ;
}
}
else if ( MtrkChunkData-> eventType[ i] == 2 ) {
printf ( "Event Parameters:\n Value: 0x" ) ; for ( j= 0 ; j< MtrkChunkData-> sysExEventValues[ i] .length ; j++ ) {
printf ( "%02X" , MtrkChunkData
-> sysExEventValues
[ i
] .
data [ j
] ) ; }
}
else {
switch ( MtrkChunkData-> midiEventValues[ i] .type ) {
case NOTE_OFF:
break ;
case NOTE_ON:
break ;
case NOTE_AFTERTOUCH:
strcpy ( str
, "Note Aftertouch Event" ) ; break ;
case CONTROLLER:
strcpy ( str
, "Controller Event" ) ; break ;
case PROGRAM_CHANGE:
strcpy ( str
, "Program Change Event" ) ; break ;
case PROGRAM_AFTERTOUCH:
strcpy ( str
, "Program Aftertouch Event" ) ; break ;
case PITCH_BEND:
strcpy ( str
, "Pitch Bend Event" ) ; break ;
default :
}
printf ( "Event Sub-Type: %26s\n " , str
) ; switch ( MtrkChunkData-> midiEventValues[ i] .type ) {
case NOTE_OFF:
case NOTE_ON:
printf ( "Event Parameters:\n " ) ; printf ( "Note Number: %d Note Velocity: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case NOTE_AFTERTOUCH:
printf ( "Event Parameters:\n " ) ; printf ( "Note Number: %d Aftertouch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case CONTROLLER:
printf ( "Event Parameters:\n " ) ; printf ( "Controller Number: %d Controller Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case PROGRAM_CHANGE:
printf ( "Event Parameters:\n " ) ; printf ( "Program Number: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 ) ; break ;
case PROGRAM_AFTERTOUCH:
printf ( "Event Parameters:\n " ) ; printf ( "Aftertouch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 ) ; break ;
case PITCH_BEND:
printf ( "Event Parameters:\n " ) ; printf ( "Pitch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 + ( MtrkChunkData
-> midiEventValues
[ i
] .
param2 >> 8 ) ) ; break ;
default :
break ;
}
}
printf ( "Press ANY key to continue...\n " ) ; }
}
return 1 ;
}
int _ParseMidi( char * fileName, int print, int save) {
FILE * midiFile;
_MthdCD
* MthdData
= ( _MthdCD
* ) ( malloc ( sizeof ( _MthdCD
) ) ) ; midiFile
= fopen ( fileName
, "rb" ) ; if ( midiFile == NULL) {
printf ( "Failed to open the file. Are you sure the file exists?\n " ) ; return 0 ;
}
if ( ! _ReadMthdChunk( midiFile, MthdData, print) ) {
printf ( "Error reading MThd Chunk. Bad MThd.\n " ) ; return 0 ;
}
_MtrkCD
** MtrkDatas
= ( _MtrkCD
** ) ( malloc ( sizeof ( _MtrkCD
* ) ) ) ; uint64_t i, j, k, p;
for ( p= 0 ; p< MthdData-> tracks_count; p++ ) {
MtrkDatas
[ p
] = ( _MtrkCD
* ) ( malloc ( sizeof ( _MtrkCD
) ) ) ; if ( ! _ReadMtrkChunk( midiFile, MtrkDatas[ p] , p, print) ) {
printf ( "Error reading MTrk Chunk. Bad MTrk.\n " ) ; return 0 ;
}
}
if ( save) {
char saveFileName[ 40 ] ;
strcpy ( saveFileName
, fileName
) ; FILE * saveFile;
//printf("saveFileName: %s.",saveFileName);
saveFile
= fopen ( saveFileName
, "w" ) ; if ( saveFile == NULL) {
printf ( "Failed to open save file.\n " ) ; return 0 ;
}
fprintf ( saveFile
, "\n \n \n --------------------------------------------------------------------------------\n \n \n \n Parsed MThd:\n \n " ) ; fprintf ( saveFile
, "Chunk Name: %26sMThd\n " , "" ) ; fprintf ( saveFile
, "Chunk Size: %23s6 bytes\n \n " , "" ) ; char str[ 40 ] ;
switch ( MthdData-> format) {
case 0x00000000 :
break ;
case 0x00000001 :
break ;
case 0x00000002 :
break ;
default :
return 0 ;
}
fprintf ( saveFile
, "Midi Format: %29s\n " , str
) ; fprintf ( saveFile
, "Number of Tracks: %24d\n " , MthdData
-> tracks_count
) ; switch ( MthdData-> time_divisionType) {
case 0x00 :
strcpy ( str
, "Ticks per Quarter Note" ) ; break ;
case 0x01 :
strcpy ( str
, "Frames per Second" ) ; break ;
default :
return 0 ;
}
fprintf ( saveFile
, "Time Division Type: %22s\n Time Division Value: %21d\n " , str
, MthdData
-> time_division
) ;
for ( k= 0 ; k< ( MthdData-> tracks_count) ; k++ ) {
_MtrkCD * MtrkChunkData = MtrkDatas[ k] ;
fprintf ( saveFile
, "\n \n \n --------------------------------------------------------------------------------\n " ) ; fprintf ( saveFile
, "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n " ) ; fprintf ( saveFile
, "--------------------------------------------------------------------------------\n \n \n \n Parsed MTrk[%d]:\n \n " , k
) ; fprintf ( saveFile
, "Chunk Name: %26sMTrk\n " , "" ) ; fprintf ( saveFile
, "Chunk Size: %24d bytes\n " , MtrkChunkData
-> byteCount
) ; fprintf ( saveFile
, "Number of Events: %24d\n " , MtrkChunkData
-> elementCount
) ; for ( i= 0 ; i< MtrkChunkData-> elementCount; i++ ) {
fprintf ( saveFile
, "\n Delta Time Value: %18d ticks\n " , ( MtrkChunkData
-> deltaTime
) [ i
] ) ; switch ( ( MtrkChunkData-> eventType) [ i] ) {
case 2 :
strcpy ( str
, "System Exclusive Event" ) ; break ;
case 1 :
break ;
case 0 :
break ;
}
fprintf ( saveFile
, "Event Type: %30s\n " , str
) ; if ( MtrkChunkData-> eventType[ i] == 1 ) {
switch ( ( ( MtrkChunkData-> metaEventValues) [ i] ) .type ) {
case SEQUENCE_NUMBER:
strcpy ( str
, "Sequence Number Event" ) ; break ;
case TEXT:
break ;
case COPYRIGHT_NOTICE:
strcpy ( str
, "Copyright Notice Event" ) ; break ;
case SEQUENCE_TRACK_NAME:
strcpy ( str
, "Sequence / Track Name Event" ) ; break ;
case INSTRUMENT_NAME:
strcpy ( str
, "Instrument Name Event" ) ; break ;
case LYRICS:
break ;
case MARKER:
break ;
case CUE_POINT:
strcpy ( str
, "Cue Point Event" ) ; break ;
case 0x09 :
strcpy ( str
, "Unknown Text Type Event" ) ; break ;
case MIDI_CHANNEL_PREFIX:
strcpy ( str
, "MIDI Channel Prefix Event" ) ; break ;
case END_OF_TRACK:
strcpy ( str
, "End of Track Event" ) ; break ;
case SET_TEMPO:
strcpy ( str
, "Set Tempo Event" ) ; break ;
case SMPTE_OFFSET:
strcpy ( str
, "SMPTE Offset Event" ) ; break ;
case TIME_SIGNATURE:
strcpy ( str
, "Time Signature Event" ) ; break ;
case KEY_SIGNATURE:
strcpy ( str
, "Key Signature Event" ) ; break ;
case SEQUENCE_SPECIFIC:
strcpy ( str
, "Sequencer Specific Event" ) ; break ;
default :
}
fprintf ( saveFile
, "Event Sub-Type: %26s\n " , str
) ; switch ( MtrkChunkData-> metaEventValues[ i] .type ) {
case SEQUENCE_NUMBER:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Value: %0x04X\n " , ( ( ( ( ( MtrkChunkData
-> metaEventValues
) [ i
] ) .
data ) [ 0 ] ) << 8 ) + ( ( ( ( MtrkChunkData
-> metaEventValues
) [ i
] ) .
data ) [ 1 ] ) ) ; break ;
case TEXT:
case COPYRIGHT_NOTICE:
case SEQUENCE_TRACK_NAME:
case INSTRUMENT_NAME:
case LYRICS:
case MARKER:
case CUE_POINT:
case 0x09 :
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "%s\n " , ( char * ) ( ( ( MtrkChunkData
-> metaEventValues
) [ i
] ) .
data ) ) ; break ;
case MIDI_CHANNEL_PREFIX:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Channel: %d\n " , ( ( ( ( MtrkChunkData
-> metaEventValues
) [ i
] ) .
data ) [ 0 ] ) ) ; break ;
case SET_TEMPO:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Value: %8u Microseconds per Quarter Note\n " , ( ( MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] ) << 16 ) + ( ( MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] ) << 8 ) + ( MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] ) ) ; break ;
case SMPTE_OFFSET:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Hour: %-2d Min: %-2d Sec: %-2d Frames: %-2d Sub-Frames: %-2d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 3 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 4 ] ) ; break ;
case TIME_SIGNATURE:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Numerator: %-3d Denominator: %-3d Metronome: %-3d 1/32: %-3d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 2 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 3 ] ) ; break ;
case KEY_SIGNATURE:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Key: %+2d Scale: %d\n " , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 0 ] , MtrkChunkData
-> metaEventValues
[ i
] .
data [ 1 ] ) ; break ;
default :
break ;
}
}
else if ( MtrkChunkData-> eventType[ i] == 2 ) {
fprintf ( saveFile
, "Event Parameters:\n Value: 0x" ) ; for ( j= 0 ; j< ( ( ( MtrkChunkData-> sysExEventValues) [ i] ) .length ) ; j++ ) {
fprintf ( saveFile
, "%02X" , MtrkChunkData
-> sysExEventValues
[ i
] .
data [ j
] ) ; }
}
else {
switch ( MtrkChunkData-> midiEventValues[ i] .type ) {
case NOTE_OFF:
break ;
case NOTE_ON:
break ;
case NOTE_AFTERTOUCH:
strcpy ( str
, "Note Aftertouch Event" ) ; break ;
case CONTROLLER:
strcpy ( str
, "Controller Event" ) ; break ;
case PROGRAM_CHANGE:
strcpy ( str
, "Program Change Event" ) ; break ;
case PROGRAM_AFTERTOUCH:
strcpy ( str
, "Program Aftertouch Event" ) ; break ;
case PITCH_BEND:
strcpy ( str
, "Pitch Bend Event" ) ; break ;
default :
}
fprintf ( saveFile
, "Event Sub-Type: %26s\n " , str
) ; switch ( MtrkChunkData-> midiEventValues[ i] .type ) {
case NOTE_OFF:
case NOTE_ON:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Note Number: %d Note Velocity: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case NOTE_AFTERTOUCH:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Note Number: %d Aftertouch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case CONTROLLER:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Controller Number: %d Controller Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 , MtrkChunkData
-> midiEventValues
[ i
] .
param2 ) ; break ;
case PROGRAM_CHANGE:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Program Number: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 ) ; break ;
case PROGRAM_AFTERTOUCH:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Aftertouch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 ) ; break ;
case PITCH_BEND:
fprintf ( saveFile
, "Event Parameters:\n " ) ; fprintf ( saveFile
, "Pitch Value: %d\n " , MtrkChunkData
-> midiEventValues
[ i
] .
param1 + ( MtrkChunkData
-> midiEventValues
[ i
] .
param2 >> 8 ) ) ; break ;
default :
break ;
}
}
}
}
fprintf ( saveFile
, "\n \n \n --------------------------------------------------------------------------------\n \n \n " ) ; fprintf ( saveFile
, "Additional Information:\n " ) ; uint64_t totalSize= 0 ;
totalSize += 14 ; //MThd has fixed size of 14 bytes
for ( k= 0 ; k< MthdData-> tracks_count; k++ ) {
_MtrkCD * MtrkChunkData = MtrkDatas[ k] ;
totalSize += MtrkChunkData-> byteCount;
}
fprintf ( saveFile
, "\n File is OK!\n Total: %u bytes\n " , totalSize
) ; }
return 1 ;
}
int main( ) {
// use all variables as unsigned as long as it's feasible, otherwise you will risk wasting at least one bit of data os Sig
char fileName[ 40 ] ;
int opt;
int ret;
scanf ( "%[^\n ]39s" , fileName
) ; printf ( "1- Check File\n 2- Parse File\n 3- Save File\n " ) ; switch ( opt) {
case 1 :
ret = _ParseMidi( fileName, 0 , 0 ) ;
break ;
case 2 :
ret = _ParseMidi( fileName, 1 , 0 ) ;
break ;
case 3 :
ret = _ParseMidi( fileName, 0 , 1 ) ;
break ;
default :
break ;
}
if ( ! ret) {
printf ( "File has error(s).\n " ) ; }
else {
}
printf ( "Press ENTER to exit the application.\n " ) ; return 0 ;
}
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <stdint.h>
#include <string.h>

//midi event types
enum{
	NOTE_OFF 				= 0x8,
	NOTE_ON 				= 0x9,
	NOTE_AFTERTOUCH 		= 0xA,
	CONTROLLER 				= 0xB,
	PROGRAM_CHANGE 			= 0xC,
	PROGRAM_AFTERTOUCH 		= 0xD,
	PITCH_BEND 				= 0xE,
};

//meta event types
enum{
	SEQUENCE_NUMBER 		= 0x00,
	TEXT 					= 0x01,
	COPYRIGHT_NOTICE 		= 0x02,
	SEQUENCE_TRACK_NAME 	= 0x03,
	INSTRUMENT_NAME 		= 0x04,
	LYRICS 					= 0x05,
	MARKER 					= 0x06,
	CUE_POINT 				= 0x07,
	//UNKNOWN_TEXT			= 0x09;
	MIDI_CHANNEL_PREFIX 	= 0x20,
	END_OF_TRACK 			= 0x2F,
	SET_TEMPO 				= 0x51,
	SMPTE_OFFSET 			= 0x54,
	TIME_SIGNATURE 			= 0x58,
	KEY_SIGNATURE			= 0x59,
	SEQUENCE_SPECIFIC 		= 0x7F,
};

//system exclusive events
/*enum{
	NORMAL = 0xF0;
	DIVIDED = 0xF0;
	DIVIDED = 0xF7;
};*/

typedef struct _MthdCD{
	uint16_t format;
	uint16_t tracks_count;
	uint16_t time_division;
	uint8_t time_divisionType;
}_MthdCD;

typedef struct _metaEventStruct{
	uint8_t type;
	uint64_t length;
	uint8_t *data;
}_metaEventStruct;

typedef struct _midiEventStruct{
	uint8_t type;
	uint8_t channel;
	uint8_t param1, param2;
}_midiEventStruct;

typedef struct _sysExEventStruct{
	uint64_t length;
	uint8_t *data;
}_sysExEventStruct;

typedef struct _MtrkCD{
	uint32_t byteCount;
	uint64_t elementCount;
	uint64_t *deltaTime;
	uint8_t *eventType;
	_metaEventStruct *metaEventValues;
	_midiEventStruct *midiEventValues;
	_sysExEventStruct *sysExEventValues;
}_MtrkCD;

uint64_t _ReadVariableLength(FILE *file,int *l);
uint8_t _ReadByte(FILE *file);
uint16_t _ReadByte2(FILE *file);
uint32_t _ReadByte4(FILE *file);

uint8_t _firstBits4(int byte);
uint8_t _lastBits4(int byte);
uint8_t _firstByte(int bytes);
uint8_t _lastByte(int bytes);

int _ReadMthdChunk(FILE *file, _MthdCD *MthdChunkData, int print);
int _ReadMtrkChunk(FILE *file, _MtrkCD *MtrkChunkData, int trackID, int print);

int _ParseMidi(char *fileName, int print, int save);

uint64_t _ReadVariableLength(FILE *file, int *l){
	*l=0;
	uint64_t totalVal = 0;
	uint8_t curVal=0, copyVal=0;
	do{
		copyVal = curVal = _ReadByte(file);
		copyVal &= ~(1 << 7);
		totalVal = (totalVal << 7) + copyVal;
		(*l)++;
	}while(curVal&(1<<7));
	return totalVal;
}

uint8_t _ReadByte(FILE *file){
	return (fgetc(file));
}

uint16_t _ReadByte2(FILE *file){
	uint16_t res = fgetc(file);
	res = (res<<8) + fgetc(file);
	return res;
}

uint32_t _ReadByte4(FILE *file){
	uint32_t res = fgetc(file);
	res = (res<<8) + fgetc(file);
	res = (res<<8) + fgetc(file);
	res = (res<<8) + fgetc(file);
	return res;
}

uint8_t _firstBits4(int byte){
	return (byte>>4);
}

uint8_t _lastBits4(int byte){
	return byte-((byte>>4)<<4);
}
uint8_t _firstByte(int bytes){
	return (bytes>>8);
}

uint8_t _lastByte(int bytes){
	return bytes-((bytes>>8)<<8);
}

int _ReadMthdChunk(FILE *file, _MthdCD *MthdChunkData, int print){
	uint32_t chkName = _ReadByte4(file);
	if (chkName != 0x4D546864){
		return 0;
	}
	uint32_t chkLength = _ReadByte4(file);
	if (chkLength != 0x00000006){
		return 0;
	}
	MthdChunkData->format = _ReadByte2(file);
	MthdChunkData->tracks_count = _ReadByte2(file);
	MthdChunkData->time_division = _ReadByte2(file);
	if (MthdChunkData->time_divisionType & (1<<16)){
		MthdChunkData->time_divisionType = 1;
	}
	else{
		MthdChunkData->time_divisionType = 0;
	}
	if (print){
		printf("\n\n\n--------------------------------------------------------------------------------\nParsed MThd:\n\n");
		printf("Chunk Name: %26s%c%c%c%c\n","",(char)(chkName>>24),(char)((chkName>>16)%0x100),(char)((chkName>>8)%0x100),(char)((chkName)%0x100));
		printf("Chunk Size: %24d bytes\n\n",chkLength);
		char str[25];
		switch (MthdChunkData->format){
			case 0x00000000:
				strcpy(str,"Single Track");
				break;
			case 0x00000001:
				strcpy(str,"Multiple Track");
				break;
			case 0x00000002:
				strcpy(str,"Multiple Song");
				break;
			default:
				return 0;
		}
		printf("Midi Format: %29s\n",str);
		printf("Number of Tracks: %24d\n",MthdChunkData->tracks_count);
		switch(MthdChunkData->time_divisionType){
			case 0x00:
				strcpy(str,"Ticks per Quarter Note");
				break;
			case 0x01:
				strcpy(str,"Frames per Second");
				break;
			default:
				return 0;
		}
		printf("Time Division Type: %22s\nTime Division Value: %21d\n",str,MthdChunkData->time_division);
	}
	return 1;
}

int _ReadMtrkChunk(FILE *file, _MtrkCD *MtrkChunkData, int trackID, int print){
	//static int trackID = 0;
	int i,j;
	uint32_t chkName = _ReadByte4(file);
	if (chkName != 0x4D54726B){
		return 0;
	}
	MtrkChunkData->byteCount = _ReadByte4(file);
	uint32_t parsedLength = MtrkChunkData->byteCount;
	
	uint8_t midiEventType;
	uint8_t midiChannel;
	
	int l=0;
	
	MtrkChunkData->elementCount=0;
	MtrkChunkData->deltaTime=(uint64_t *)(malloc(sizeof(uint64_t)));
	MtrkChunkData->eventType=(uint8_t *)(malloc(sizeof(uint8_t)));
	MtrkChunkData->metaEventValues=(_metaEventStruct *)(malloc(sizeof(_metaEventStruct)));
	MtrkChunkData->midiEventValues=(_midiEventStruct *)(malloc(sizeof(_midiEventStruct)));
	MtrkChunkData->sysExEventValues=(_sysExEventStruct *)(malloc(sizeof(_sysExEventStruct)));
	
	while (parsedLength > 0){
		MtrkChunkData->deltaTime=(realloc(MtrkChunkData->deltaTime,sizeof(uint64_t) * (MtrkChunkData->elementCount+1)));
		MtrkChunkData->eventType=(realloc(MtrkChunkData->eventType,sizeof(uint8_t) * (MtrkChunkData->elementCount+1)));
		MtrkChunkData->metaEventValues=(realloc(MtrkChunkData->metaEventValues,sizeof(_metaEventStruct) * (MtrkChunkData->elementCount+1)));
		MtrkChunkData->midiEventValues=(realloc(MtrkChunkData->midiEventValues,sizeof(_midiEventStruct) * (MtrkChunkData->elementCount+1)));
		MtrkChunkData->sysExEventValues=(realloc(MtrkChunkData->sysExEventValues,sizeof(_sysExEventStruct) * (MtrkChunkData->elementCount+1)));
		
		//getch();
		//printf("looped... toBeParsedLength is %d \n",parsedLength);
		((MtrkChunkData->deltaTime)[MtrkChunkData->elementCount]) = _ReadVariableLength(file,&l);
		//printf("Variable length: %d\n",MtrkChunkData->deltaTime);
		parsedLength-=l;
		uint8_t nByte = _ReadByte(file);
		//printf("nByte: 0x%X\n",nByte);
		parsedLength-=1;
		//check nByte for event type
		switch(nByte){
			//meta event case
			case 0xFF:
				((MtrkChunkData->eventType)[MtrkChunkData->elementCount])=1;
				//printf("Found META event\n");
				((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).type = _ReadByte(file);
				//printf("Meta event 0x%X\n",metaEventType);
				parsedLength-=1;
				switch(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).type){
					//might seem trivial to seperate types since all is scanned the same way but it is actually needed for separating conversion and track combination
					case SEQUENCE_NUMBER:
					case KEY_SIGNATURE:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadByte(file);
						parsedLength-=1;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case SEQUENCE_SPECIFIC:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case TEXT:
					case COPYRIGHT_NOTICE:
					case SEQUENCE_TRACK_NAME:
					case INSTRUMENT_NAME:
					case LYRICS:
					case MARKER:
					case CUE_POINT:
					case 0x09: //unknown text event
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i]='\0';
						break;
					case MIDI_CHANNEL_PREFIX:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case SET_TEMPO:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case SMPTE_OFFSET:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case TIME_SIGNATURE:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					case END_OF_TRACK:
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
						parsedLength-=l;
						((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length));
						for (i=0;i<((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).length;i++){
							(((MtrkChunkData->metaEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
							parsedLength-=1;
						}
						break;
					default:
						return 0;
				}
				break;
			//system exclusive event case
			case 0xF0:
			case 0xF7:
				((MtrkChunkData->eventType)[MtrkChunkData->elementCount])=2;
				//printf("Found System Exclusive event\n");
				((MtrkChunkData->sysExEventValues)[MtrkChunkData->elementCount]).length=_ReadVariableLength(file,&l);
				parsedLength-=l;
				((MtrkChunkData->sysExEventValues)[MtrkChunkData->elementCount]).data=(uint8_t *)(malloc(sizeof(uint8_t) * ((MtrkChunkData->sysExEventValues)[MtrkChunkData->elementCount]).length));
				for (i=0;i<((MtrkChunkData->sysExEventValues)[MtrkChunkData->elementCount]).length;i++){
					(((MtrkChunkData->sysExEventValues)[MtrkChunkData->elementCount]).data)[i] = _ReadByte(file);
					parsedLength-=1;
				}
				break;
			//midi event case
			default:
				((MtrkChunkData->eventType)[MtrkChunkData->elementCount])=0;
				//printf("Found MIDI event\n");
				midiEventType = _firstBits4(nByte);
				midiChannel = _lastBits4(nByte);
				//printf("MIDI event type: 0x%X\n",midiEventType);
				switch (midiEventType){
					case NOTE_OFF:
					case NOTE_ON:
					case NOTE_AFTERTOUCH:
					case CONTROLLER:
					case PITCH_BEND:
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).type = midiEventType;
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).channel = midiChannel;
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).param1 = _ReadByte(file);
						parsedLength-=1;
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount].param2) = _ReadByte(file);
						parsedLength-=1;
						break;
					case PROGRAM_CHANGE:
					case PROGRAM_AFTERTOUCH:
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).type = midiEventType;
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).channel = midiChannel;
						((MtrkChunkData->midiEventValues)[MtrkChunkData->elementCount]).param1 = _ReadByte(file);
						parsedLength-=1;
						break;
					default:
						return 0;
				}
				break;
		}
		(MtrkChunkData->elementCount)++;
	}
	if (print){
		printf("\n\n\n--------------------------------------------------------------------------------\n");
		printf("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n");
		printf("--------------------------------------------------------------------------------\nParsed MTrk[%d]:\n\n",trackID);
		printf("Chunk Name: %26s%c%c%c%c\n","",(char)(chkName>>24),(char)((chkName>>16)%0x100),(char)((chkName>>8)%0x100),(char)((chkName)%0x100));
		printf("Chunk Size: %24d bytes\n",MtrkChunkData->byteCount);
		printf("Number of Events: %24d\n",MtrkChunkData->elementCount);
		printf("\nEvents:\n");
		char str[40]={'\0'};
		for (i=0;i<MtrkChunkData->elementCount;i++){
			printf("\nDelta Time Value: %18d ticks\n",MtrkChunkData->deltaTime[i]);
			switch(MtrkChunkData->eventType[i]){
				case 2:
					strcpy(str,"System Exclusive Event");
					break;
				case 1:
					strcpy(str,"META Event");
					break;
				case 0:
					strcpy(str,"MIDI Event");
					break;
			}
			printf("Event Type: %30s\n",str);
			if (MtrkChunkData->eventType[i] == 1){
				switch(MtrkChunkData->metaEventValues[i].type){
					case SEQUENCE_NUMBER:
						strcpy(str,"Sequence Number Event");
						break;
					case TEXT:
						strcpy(str,"Text Event");
						break;
					case COPYRIGHT_NOTICE:
						strcpy(str,"Copyright Notice Event");
						break;
					case SEQUENCE_TRACK_NAME:
						strcpy(str,"Sequence / Track Name Event");
						break;
					case INSTRUMENT_NAME:
						strcpy(str,"Instrument Name Event");
						break;
					case LYRICS:
						strcpy(str,"Lyrics Event");
						break;
					case MARKER:
						strcpy(str,"Marker Event");
						break;
					case CUE_POINT:
						strcpy(str,"Cue Point Event");
						break;
					case 0x09:
						strcpy(str,"Unknown Text Type Event");
						break;
					case MIDI_CHANNEL_PREFIX:
						strcpy(str,"MIDI Channel Prefix Event");
						break;
					case END_OF_TRACK:
						strcpy(str,"End of Track Event");
						break;
					case SET_TEMPO:
						strcpy(str,"Set Tempo Event");
						break;
					case SMPTE_OFFSET:
						strcpy(str,"SMPTE Offset Event");
						break;
					case TIME_SIGNATURE:
						strcpy(str,"Time Signature Event");
						break;
					case KEY_SIGNATURE:
						strcpy(str,"Key Signature Event");
						break;
					case SEQUENCE_SPECIFIC:
						strcpy(str,"Sequencer Specific Event");
						break;
					default:
						strcpy(str,"UNKNOWN EVENT");
				}
				printf("Event Sub-Type: %26s\n",str);
				switch(MtrkChunkData->metaEventValues[i].type){
					case SEQUENCE_NUMBER:
						printf("Event Parameters:\n");
						printf("Value: %0x04X\n",MtrkChunkData->metaEventValues[i].data[0]<<8+MtrkChunkData->metaEventValues[i].data[1]);
						break;
					case TEXT:
					case COPYRIGHT_NOTICE:
					case SEQUENCE_TRACK_NAME:
					case INSTRUMENT_NAME:
					case LYRICS:
					case MARKER:
					case CUE_POINT:
					case 0x09:
						printf("Event Parameters:\n");
						printf("%s\n",(char *)MtrkChunkData->metaEventValues[i].data);
						break;
					case MIDI_CHANNEL_PREFIX:
						printf("Event Parameters:\n");
						printf("Channel: %d\n",MtrkChunkData->metaEventValues[i].data[0]);
						break;
					case SET_TEMPO:
						printf("Event Parameters:\n");
						printf("Value: %8u Microseconds per Quarter Note\n",(MtrkChunkData->metaEventValues[i].data[0]<<16)+(MtrkChunkData->metaEventValues[i].data[1]<<8)+MtrkChunkData->metaEventValues[i].data[2]);
						break;
					case SMPTE_OFFSET:
						printf("Event Parameters:\n");
						printf("Hour: %-2d  Min: %-2d  Sec: %-2d  Frames: %-2d  Sub-Frames: %-2d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1],MtrkChunkData->metaEventValues[i].data[2],MtrkChunkData->metaEventValues[i].data[3],MtrkChunkData->metaEventValues[i].data[4]);
						break;
					case TIME_SIGNATURE:
						printf("Event Parameters:\n");
						printf("Numerator: %-3d  Denominator: %-3d  Metronome: %-3d  1/32: %-3d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1],MtrkChunkData->metaEventValues[i].data[2],MtrkChunkData->metaEventValues[i].data[3]);
						break;
					case KEY_SIGNATURE:
						printf("Event Parameters:\n");
						printf("Key: %+2d Scale: %d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1]);
						break;
					default:
						break;
				}
			}
			else if (MtrkChunkData->eventType[i] == 2){
				printf("Event Parameters:\n Value: 0x");
				for (j=0;j<MtrkChunkData->sysExEventValues[i].length;j++){
					printf("%02X",MtrkChunkData->sysExEventValues[i].data[j]);
				}
				printf("\n");
			}
			else{
				switch(MtrkChunkData->midiEventValues[i].type){
					case NOTE_OFF:
						strcpy(str,"Note-Off Event");
						break;
					case NOTE_ON:
						strcpy(str,"Note-On Event");
						break;
					case NOTE_AFTERTOUCH:
						strcpy(str,"Note Aftertouch Event");
						break;
					case CONTROLLER:
						strcpy(str,"Controller Event");
						break;
					case PROGRAM_CHANGE:
						strcpy(str,"Program Change Event");
						break;
					case PROGRAM_AFTERTOUCH:
						strcpy(str,"Program Aftertouch Event");
						break;
					case PITCH_BEND:
						strcpy(str,"Pitch Bend Event");
						break;
					default:
						strcpy(str,"UNKNOWN Event");
				}
				printf("Event Sub-Type: %26s\n",str);
				switch(MtrkChunkData->midiEventValues[i].type){
					case NOTE_OFF:
					case NOTE_ON:
						printf("Event Parameters:\n");
						printf("Note Number: %d  Note Velocity: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
						break;
					case NOTE_AFTERTOUCH:
						printf("Event Parameters:\n");
						printf("Note Number: %d  Aftertouch Value: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
						break;
					case CONTROLLER:
						printf("Event Parameters:\n");
						printf("Controller Number: %d  Controller Value: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
						break;
					case PROGRAM_CHANGE:
						printf("Event Parameters:\n");
						printf("Program Number: %d\n",MtrkChunkData->midiEventValues[i].param1);
						break;
					case PROGRAM_AFTERTOUCH:
						printf("Event Parameters:\n");
						printf("Aftertouch Value: %d\n",MtrkChunkData->midiEventValues[i].param1);
						break;
					case PITCH_BEND:
						printf("Event Parameters:\n");
						printf("Pitch Value: %d\n",MtrkChunkData->midiEventValues[i].param1 + (MtrkChunkData->midiEventValues[i].param2>>8));
						break;
					default:
						break;
				}
			}
			printf("Press ANY key to continue...\n");
			getch();
		}
	}
	return 1;
}

int _ParseMidi(char *fileName, int print, int save){
	FILE *midiFile;
	_MthdCD *MthdData = (_MthdCD *)(malloc(sizeof(_MthdCD)));
	midiFile = fopen(fileName, "rb");
	if (midiFile == NULL){
		printf("Failed to open the file. Are you sure the file exists?\n");
		return 0;
	}
	if (!_ReadMthdChunk(midiFile,MthdData, print)){
		printf("Error reading MThd Chunk. Bad MThd.\n");
		return 0;
	}
	
	_MtrkCD **MtrkDatas = (_MtrkCD **)(malloc(sizeof(_MtrkCD *)));
	uint64_t i,j,k,p;
	for (p=0;p<MthdData->tracks_count;p++){
		MtrkDatas[p] = (_MtrkCD *)(malloc(sizeof(_MtrkCD)));
		if(!_ReadMtrkChunk(midiFile,MtrkDatas[p],p,print)){
			printf("Error reading MTrk Chunk. Bad MTrk.\n");
			return 0;
		}
	}
	
	fclose(midiFile);
	
	if (save){
		char saveFileName[40];
		strcpy(saveFileName,fileName);
		strcpy(saveFileName+strlen(fileName)-4," Parsed.txt");
		FILE *saveFile;
		//printf("saveFileName: %s.",saveFileName);
		
		saveFile = fopen(saveFileName,"w");
		if (saveFile == NULL){
			printf("Failed to open save file.\n");
			return 0;
		}
		
		fprintf(saveFile,"\n\n\n--------------------------------------------------------------------------------\n\n\n\nParsed MThd:\n\n");
		fprintf(saveFile,"Chunk Name: %26sMThd\n","");
		fprintf(saveFile,"Chunk Size: %23s6 bytes\n\n","");
		char str[40];
		switch (MthdData->format){
			case 0x00000000:
				strcpy(str,"Single Track");
				break;
			case 0x00000001:
				strcpy(str,"Multiple Track");
				break;
			case 0x00000002:
				strcpy(str,"Multiple Song");
				break;
			default:
				return 0;
		}
		fprintf(saveFile,"Midi Format: %29s\n",str);
		fprintf(saveFile,"Number of Tracks: %24d\n",MthdData->tracks_count);
		switch(MthdData->time_divisionType){
			case 0x00:
				strcpy(str,"Ticks per Quarter Note");
				break;
			case 0x01:
				strcpy(str,"Frames per Second");
				break;
			default:
				return 0;
		}
		fprintf(saveFile,"Time Division Type: %22s\nTime Division Value: %21d\n",str,MthdData->time_division);
		
		for (k=0;k<(MthdData->tracks_count);k++){
			_MtrkCD *MtrkChunkData = MtrkDatas[k];
			fprintf(saveFile,"\n\n\n--------------------------------------------------------------------------------\n");
			fprintf(saveFile,"++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n");
			fprintf(saveFile,"--------------------------------------------------------------------------------\n\n\n\nParsed MTrk[%d]:\n\n",k);
			fprintf(saveFile,"Chunk Name: %26sMTrk\n","");
			fprintf(saveFile,"Chunk Size: %24d bytes\n",MtrkChunkData->byteCount);
			fprintf(saveFile,"Number of Events: %24d\n",MtrkChunkData->elementCount);
			fprintf(saveFile,"\nEvents:\n");
			for (i=0;i<MtrkChunkData->elementCount;i++){
				fprintf(saveFile,"\nDelta Time Value: %18d ticks\n",(MtrkChunkData->deltaTime)[i]);
				switch((MtrkChunkData->eventType)[i]){
					case 2:
						strcpy(str,"System Exclusive Event");
						break;
					case 1:
						strcpy(str,"META Event");
						break;
					case 0:
						strcpy(str,"MIDI Event");
						break;
				}
				fprintf(saveFile,"Event Type: %30s\n",str);
				if (MtrkChunkData->eventType[i] == 1){
					switch(((MtrkChunkData->metaEventValues)[i]).type){
						case SEQUENCE_NUMBER:
							strcpy(str,"Sequence Number Event");
							break;
						case TEXT:
							strcpy(str,"Text Event");
							break;
						case COPYRIGHT_NOTICE:
							strcpy(str,"Copyright Notice Event");
							break;
						case SEQUENCE_TRACK_NAME:
							strcpy(str,"Sequence / Track Name Event");
							break;
						case INSTRUMENT_NAME:
							strcpy(str,"Instrument Name Event");
							break;
						case LYRICS:
							strcpy(str,"Lyrics Event");
							break;
						case MARKER:
							strcpy(str,"Marker Event");
							break;
						case CUE_POINT:
							strcpy(str,"Cue Point Event");
							break;
						case 0x09:
							strcpy(str,"Unknown Text Type Event");
							break;
						case MIDI_CHANNEL_PREFIX:
							strcpy(str,"MIDI Channel Prefix Event");
							break;
						case END_OF_TRACK:
							strcpy(str,"End of Track Event");
							break;
						case SET_TEMPO:
							strcpy(str,"Set Tempo Event");
							break;
						case SMPTE_OFFSET:
							strcpy(str,"SMPTE Offset Event");
							break;
						case TIME_SIGNATURE:
							strcpy(str,"Time Signature Event");
							break;
						case KEY_SIGNATURE:
							strcpy(str,"Key Signature Event");
							break;
						case SEQUENCE_SPECIFIC:
							strcpy(str,"Sequencer Specific Event");
							break;
						default:
							strcpy(str,"UNKNOWN EVENT");
					}
					fprintf(saveFile,"Event Sub-Type: %26s\n",str);
					switch(MtrkChunkData->metaEventValues[i].type){
						case SEQUENCE_NUMBER:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Value: %0x04X\n",(((((MtrkChunkData->metaEventValues)[i]).data)[0])<<8)+((((MtrkChunkData->metaEventValues)[i]).data)[1]));
							break;
						case TEXT:
						case COPYRIGHT_NOTICE:
						case SEQUENCE_TRACK_NAME:
						case INSTRUMENT_NAME:
						case LYRICS:
						case MARKER:
						case CUE_POINT:
						case 0x09:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"%s\n",(char *)(((MtrkChunkData->metaEventValues)[i]).data));
							break;
						case MIDI_CHANNEL_PREFIX:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Channel: %d\n",((((MtrkChunkData->metaEventValues)[i]).data)[0]));
							break;
						case SET_TEMPO:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Value: %8u Microseconds per Quarter Note\n",((MtrkChunkData->metaEventValues[i].data[0])<<16)+((MtrkChunkData->metaEventValues[i].data[1])<<8)+(MtrkChunkData->metaEventValues[i].data[2]));
							break;
						case SMPTE_OFFSET:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Hour: %-2d  Min: %-2d  Sec: %-2d  Frames: %-2d  Sub-Frames: %-2d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1],MtrkChunkData->metaEventValues[i].data[2],MtrkChunkData->metaEventValues[i].data[3],MtrkChunkData->metaEventValues[i].data[4]);
							break;
						case TIME_SIGNATURE:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Numerator: %-3d  Denominator: %-3d  Metronome: %-3d  1/32: %-3d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1],MtrkChunkData->metaEventValues[i].data[2],MtrkChunkData->metaEventValues[i].data[3]);
							break;
						case KEY_SIGNATURE:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Key: %+2d Scale: %d\n",MtrkChunkData->metaEventValues[i].data[0],MtrkChunkData->metaEventValues[i].data[1]);
							break;
						default:
							break;
					}
				}
				else if (MtrkChunkData->eventType[i] == 2){
					fprintf(saveFile,"Event Parameters:\n Value: 0x");
					for (j=0;j<(((MtrkChunkData->sysExEventValues)[i]).length);j++){
						fprintf(saveFile,"%02X",MtrkChunkData->sysExEventValues[i].data[j]);
					}
					fprintf(saveFile,"\n");
				}
				else{
					switch(MtrkChunkData->midiEventValues[i].type){
						case NOTE_OFF:
							strcpy(str,"Note-Off Event");
							break;
						case NOTE_ON:
							strcpy(str,"Note-On Event");
							break;
						case NOTE_AFTERTOUCH:
							strcpy(str,"Note Aftertouch Event");
							break;
						case CONTROLLER:
							strcpy(str,"Controller Event");
							break;
						case PROGRAM_CHANGE:
							strcpy(str,"Program Change Event");
							break;
						case PROGRAM_AFTERTOUCH:
							strcpy(str,"Program Aftertouch Event");
							break;
						case PITCH_BEND:
							strcpy(str,"Pitch Bend Event");
							break;
						default:
							strcpy(str,"UNKNOWN Event");
					}
					fprintf(saveFile,"Event Sub-Type: %26s\n",str);
					switch(MtrkChunkData->midiEventValues[i].type){
						case NOTE_OFF:
						case NOTE_ON:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Note Number: %d  Note Velocity: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
							break;
						case NOTE_AFTERTOUCH:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Note Number: %d  Aftertouch Value: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
							break;
						case CONTROLLER:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Controller Number: %d  Controller Value: %d\n",MtrkChunkData->midiEventValues[i].param1,MtrkChunkData->midiEventValues[i].param2);
							break;
						case PROGRAM_CHANGE:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Program Number: %d\n",MtrkChunkData->midiEventValues[i].param1);
							break;
						case PROGRAM_AFTERTOUCH:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Aftertouch Value: %d\n",MtrkChunkData->midiEventValues[i].param1);
							break;
						case PITCH_BEND:
							fprintf(saveFile,"Event Parameters:\n");
							fprintf(saveFile,"Pitch Value: %d\n",MtrkChunkData->midiEventValues[i].param1 + (MtrkChunkData->midiEventValues[i].param2>>8));
							break;
						default:
							break;
					}
				}
			}
		}
		
		fprintf(saveFile,"\n\n\n--------------------------------------------------------------------------------\n\n\n");
		fprintf(saveFile,"Additional Information:\n");
		uint64_t totalSize=0;
		totalSize += 14; //MThd has fixed size of 14 bytes
		for (k=0;k<MthdData->tracks_count;k++){
			_MtrkCD *MtrkChunkData = MtrkDatas[k];
			totalSize += MtrkChunkData->byteCount;
		}
		fprintf(saveFile,"\nFile is OK!\nTotal: %u bytes\n",totalSize);
		fclose(saveFile);
	}
	return 1;
}

int main(){
	// use all variables as unsigned as long as it's feasible, otherwise you will risk wasting at least one bit of data os Sig
	char fileName[40];
	int opt;
	int ret;
	printf("Enter filename: ");
	fflush(stdin);
	scanf("%[^\n]39s",fileName);
	printf("1- Check File\n2- Parse File\n3- Save File\n");
	scanf("%d",&opt);
	switch(opt){
		case 1:
			ret = _ParseMidi(fileName,0,0);
			break;
		case 2:
			ret = _ParseMidi(fileName,1,0);
			break;
		case 3:
			ret = _ParseMidi(fileName,0,1);
			break;
		default:
			printf("Invalid input\n");
			break;
	}
	if (!ret){
		printf("File has error(s).\n");
	}
	else{
		printf("File is OK!\n");
	}
	printf("Press ENTER to exit the application.\n");
	fflush(stdin);
	gets(fileName);
	return 0;
}