diff options
Diffstat (limited to 'midifile')
-rwxr-xr-x | midifile/midifile.c | 2336 |
1 files changed, 1169 insertions, 1167 deletions
diff --git a/midifile/midifile.c b/midifile/midifile.c index 92482aa..d036bac 100755 --- a/midifile/midifile.c +++ b/midifile/midifile.c @@ -32,8 +32,6 @@ * 20060110 output totaltime before end-bang */ -//#define MSW // useful in m_pd.h if not yet replaced by _MSC_VER - #include "m_pd.h" #include <stdio.h> #include <string.h> @@ -48,50 +46,51 @@ static t_class *midifile_class; #define PATH_BUF_SIZE 256 -#define MAX_TRACKS 128 // track data is allocated as needed but we need to preallocate space for the pointers -#define MAX_TRACK_LEN 1024 // for now... +#define MAX_TRACKS 128 +/* track data is allocated as needed but we need to preallocate space for the pointers */ +#define MAX_TRACK_LEN 1024 +/* MAX_TRACK_LEN 1024 for now...*/ #define ALL_TRACKS MAX_TRACKS typedef enum {mfReset, mfReading, mfWriting} mfstate; typedef struct mf_header_chunk { - char chunk_type[4]; // each chunk begins with a 4-character ASCII type. - size_t chunk_length ; // followed by a 32-bit length - int chunk_format; - int chunk_ntrks; - int chunk_division; + char chunk_type[4]; /* each chunk begins with a 4-character ASCII type.*/ + size_t chunk_length ; /* followed by a 32-bit length */ + int chunk_format; + int chunk_ntrks; + int chunk_division; } mf_header_chunk; typedef struct mf_track_chunk { - char chunk_type[4]; // each chunk begins with a 4-character ASCII type. - size_t chunk_length ; // followed by a 32-bit length - size_t delta_time ; // current delta_time of latest track_data element - size_t total_time ; // sum of delta_times so far - size_t track_index ; // current byte offset to next track_data element - unsigned char running_status; - char *track_data; + char chunk_type[4]; /* each chunk begins with a 4-character ASCII type. */ + size_t chunk_length ; /* followed by a 32-bit length */ + size_t delta_time ; /* current delta_time of latest track_data element */ + size_t total_time ; /* sum of delta_times so far */ + size_t track_index ; /* current byte offset to next track_data element */ + unsigned char running_status; + char *track_data; } mf_track_chunk; typedef struct t_midifile { - t_object x_obj; - size_t total_time; // current time for this MIDI file in delta_time units - t_atom midi_data[3]; // one MIDI packet as a list - t_outlet *midi_list_outlet; - t_outlet *bang_outlet; - //t_outlet *sysex_outlet; // 20060105 use midi_list_outlet instead - t_outlet *total_time_outlet; - FILE *fP; - FILE *tmpFP; - char fPath[PATH_BUF_SIZE]; - size_t offset; // character offset into the file fP - int track; // play this track, or all tracks if negative; - int verbosity; // nonzero for text output to console - int ended; // nonzero if all tracks have finished - mfstate state; // READING or WRITING - mf_header_chunk header_chunk; // First chunk in the midi file - mf_track_chunk track_chunk[MAX_TRACKS]; // Subsequent track chunks. Other kinds of chunk are ignored + t_object x_obj; + size_t total_time; /* current time for this MIDI file in delta_time units */ + t_atom midi_data[3]; /* one MIDI packet as a list */ + t_outlet *midi_list_outlet; + t_outlet *bang_outlet; + t_outlet *total_time_outlet; + FILE *fP; + FILE *tmpFP; + char fPath[PATH_BUF_SIZE]; + size_t offset; /* character offset into the file fP */ + int track; /* play this track, or all tracks if negative; */ + int verbosity; /* nonzero for text output to console */ + int ended; /* nonzero if all tracks have finished */ + mfstate state; /* READING or WRITING */ + mf_header_chunk header_chunk; /* First chunk in the midi file */ + mf_track_chunk track_chunk[MAX_TRACKS]; /* Subsequent track chunks. Other kinds of chunk are ignored. */ } t_midifile; void midifile_skip_next_track_chunk_data(t_midifile *x, int index); @@ -130,111 +129,107 @@ EXPORT_SHARED void midifile_setup(void); EXPORT_SHARED void midifile_setup(void) { - midifile_class = class_new (gensym("midifile"), - (t_newmethod) midifile_new, - (t_method)midifile_free, sizeof(t_midifile), - CLASS_DEFAULT, - A_GIMME, 0); - //CLASS_DEFAULT a normal object with one inlet - class_addbang(midifile_class, midifile_bang); - class_addfloat(midifile_class, midifile_float); - class_addlist(midifile_class, midifile_list); - class_addmethod(midifile_class, (t_method)midifile_read, gensym("read"), A_DEFSYMBOL, 0); - class_addmethod(midifile_class, (t_method)midifile_flush, gensym("flush"), 0); - class_addmethod(midifile_class, (t_method)midifile_write, gensym("write"), A_DEFSYMBOL, 0); - class_addmethod(midifile_class, (t_method)midifile_dump, gensym("dump"), A_DEFFLOAT, 0); - class_addmethod(midifile_class, (t_method)midifile_single_track, gensym("track"), A_DEFFLOAT, 0); - class_addmethod(midifile_class, (t_method)midifile_rewind, gensym("rewind"), 0); - class_addmethod(midifile_class, (t_method)midifile_verbosity, gensym("verbose"), A_DEFFLOAT, 0); - class_sethelpsymbol(midifile_class, gensym("midifile-help")); + midifile_class = class_new (gensym("midifile"), + (t_newmethod) midifile_new, + (t_method)midifile_free, sizeof(t_midifile), + CLASS_DEFAULT, + A_GIMME, 0); + + class_addbang(midifile_class, midifile_bang); + class_addfloat(midifile_class, midifile_float); + class_addlist(midifile_class, midifile_list); + class_addmethod(midifile_class, (t_method)midifile_read, gensym("read"), A_DEFSYMBOL, 0); + class_addmethod(midifile_class, (t_method)midifile_flush, gensym("flush"), 0); + class_addmethod(midifile_class, (t_method)midifile_write, gensym("write"), A_DEFSYMBOL, 0); + class_addmethod(midifile_class, (t_method)midifile_dump, gensym("dump"), A_DEFFLOAT, 0); + class_addmethod(midifile_class, (t_method)midifile_single_track, gensym("track"), A_DEFFLOAT, 0); + class_addmethod(midifile_class, (t_method)midifile_rewind, gensym("rewind"), 0); + class_addmethod(midifile_class, (t_method)midifile_verbosity, gensym("verbose"), A_DEFFLOAT, 0); + class_sethelpsymbol(midifile_class, gensym("midifile-help")); } void *midifile_new(t_symbol *s, int argc, t_atom *argv) { - t_midifile *x = (t_midifile *)pd_new(midifile_class); - t_symbol *pathSymbol; - int i; - - if (x == NULL) - { - error("midifile: Could not create..."); - return x; - } - x->fP = NULL; - x->fPath[0] = '\0'; - x->track = ALL_TRACKS; // startup playing anything - x->midi_data[0].a_type = x->midi_data[1].a_type = x->midi_data[2].a_type = A_FLOAT; - x->state = mfReset; - x->verbosity = 1; // default to posting all - for (i = 0; i < MAX_TRACKS; ++i) - { - x->track_chunk[i].track_data = NULL; - } - // find the first string in the arg list and interpret it as a path to a midi file - for (i = 0; i < argc; ++i) - { - if (argv[i].a_type == A_SYMBOL) - { - pathSymbol = atom_getsymbol(&argv[i]); - if (pathSymbol != NULL) - { - if (midifile_open_path(x, pathSymbol->s_name, "rb")) - { - if (x->verbosity) post("midifile: opened %s", x->fPath); - x->state = mfReading; - if (midifle_read_chunks(x) == 0) midifile_free_file(x); - } - else error("midifile: unable to open %s", pathSymbol->s_name); - break; - } - } - } - x->midi_list_outlet = outlet_new(&x->x_obj, &s_list); - // x->sysex_outlet = outlet_new(&x->x_obj, &s_float); // raw sysex stream (20060105 send a list through midi_list_outlet) - x->total_time_outlet = outlet_new(&x->x_obj, &s_float); // current total_time - x->bang_outlet = outlet_new(&x->x_obj, &s_bang); // bang at end of file - return (void *)x; + t_midifile *x = (t_midifile *)pd_new(midifile_class); + t_symbol *pathSymbol; + int i; + + if (x == NULL) + { + error("midifile: Could not create..."); + return x; + } + x->fP = NULL; + x->fPath[0] = '\0'; + x->track = ALL_TRACKS; /* startup playing anything */ + x->midi_data[0].a_type = x->midi_data[1].a_type = x->midi_data[2].a_type = A_FLOAT; + x->state = mfReset; + x->verbosity = 1; /* default to posting all */ + for (i = 0; i < MAX_TRACKS; ++i) + { + x->track_chunk[i].track_data = NULL; + } + /* find the first string in the arg list and interpret it as a path to a midi file */ + for (i = 0; i < argc; ++i) + { + if (argv[i].a_type == A_SYMBOL) + { + pathSymbol = atom_getsymbol(&argv[i]); + if (pathSymbol != NULL) + { + if (midifile_open_path(x, pathSymbol->s_name, "rb")) + { + if (x->verbosity) post("midifile: opened %s", x->fPath); + x->state = mfReading; + if (midifle_read_chunks(x) == 0) midifile_free_file(x); + } + else error("midifile: unable to open %s", pathSymbol->s_name); + break; + } + } + } + x->midi_list_outlet = outlet_new(&x->x_obj, &s_list); + x->total_time_outlet = outlet_new(&x->x_obj, &s_float); /* current total_time */ + x->bang_outlet = outlet_new(&x->x_obj, &s_bang); /* bang at end of file */ + return (void *)x; } void midifile_close(t_midifile *x) { - if (x->fP != NULL) - { - fclose (x->fP); - x->fP = NULL; - } - if (x->tmpFP != NULL) - { - fclose(x->tmpFP); - x->tmpFP = NULL; - } - x->fPath[0] = '\0'; - x->state = mfReset; - x->total_time = 0L; - x->offset = 0L; - outlet_float(x->total_time_outlet, x->total_time); + if (x->fP != NULL) + { + fclose (x->fP); + x->fP = NULL; + } + if (x->tmpFP != NULL) + { + fclose(x->tmpFP); + x->tmpFP = NULL; + } + x->fPath[0] = '\0'; + x->state = mfReset; + x->total_time = 0L; + x->offset = 0L; + outlet_float(x->total_time_outlet, x->total_time); } void midifile_free_file(t_midifile *x) { - int i; - - midifile_close(x); - - for (i = 0; i < MAX_TRACKS; ++i) - { - if (x->track_chunk[i].track_data != NULL) - { - freebytes(x->track_chunk[i].track_data, x->track_chunk[i].chunk_length); - //if (x->verbosity) post("midifile_free: freed %lu bytes for track %d", x->track_chunk[i].chunk_length, i); - } - x->track_chunk[i].track_data = NULL; - } + int i; + + midifile_close(x); + + for (i = 0; i < MAX_TRACKS; ++i) + { + if (x->track_chunk[i].track_data != NULL) + freebytes(x->track_chunk[i].track_data, x->track_chunk[i].chunk_length); + x->track_chunk[i].track_data = NULL; + } } void midifile_free(t_midifile *x) { - midifile_free_file(x); + midifile_free_file(x); } int midifile_open_path(t_midifile *x, char *path, char *mode) @@ -244,166 +239,172 @@ int midifile_open_path(t_midifile *x, char *path, char *mode) /* midifile_open_path attempts to open the file for binary mode reading. */ /* Returns 1 if successful, else 0. */ { - FILE *fP; - char tryPath[PATH_BUF_SIZE]; - - strncpy(tryPath, path, PATH_BUF_SIZE-1); // copy path into a length-limited buffer - // ...if it doesn't work we won't mess up x->fPath - tryPath[PATH_BUF_SIZE-1] = '\0'; // just make sure there is a null termination - - fP = fopen(tryPath, mode); - if (fP == NULL) return 0; - x->fP = fP; - strncpy(x->fPath, tryPath, PATH_BUF_SIZE); - return 1; + FILE *fP; + char tryPath[PATH_BUF_SIZE]; + + strncpy(tryPath, path, PATH_BUF_SIZE-1); /* copy path into a length-limited buffer */ + /* ...if it doesn't work we won't mess up x->fPath */ + tryPath[PATH_BUF_SIZE-1] = '\0'; /* just make sure there is a null termination */ + + fP = fopen(tryPath, mode); + if (fP == NULL) return 0; + x->fP = fP; + strncpy(x->fPath, tryPath, PATH_BUF_SIZE); + return 1; } void midifile_flush(t_midifile *x) /* write the header to x->fP, copy x->tmpFP into it, and close both files */ /* flush ends the track */ { - size_t written = 0L; - size_t end_time = x->total_time; - int c; + size_t written = 0L; + size_t end_time = x->total_time; + int c; - if(x->state != mfWriting) return; // only if we're writing + if(x->state != mfWriting) return; /* only if we're writing */ - outlet_bang(x->bang_outlet); // bang so tick count can be saved externally - midifile_write_end_of_track(x, end_time); - written = midifile_write_header(x); + outlet_bang(x->bang_outlet); /* bang so tick count can be saved externally */ + midifile_write_end_of_track(x, end_time); + written = midifile_write_header(x); /* now copy the MIDI data from tmpFP to fP */ - rewind (x->tmpFP); - while ((c = getc(x->tmpFP)) != EOF) - { - putc(c, x->fP); - ++written; - } - if (x->verbosity) post ("midifile: wrote %lu to %s", written, x->fPath); - midifile_close(x); + rewind (x->tmpFP); + while ((c = getc(x->tmpFP)) != EOF) + { + putc(c, x->fP); + ++written; + } + if (x->verbosity) post ("midifile: wrote %lu to %s", written, x->fPath); + midifile_close(x); } size_t midifile_write_header(t_midifile *x) /* write the MThd and MTrk headers to x->fP */ { - size_t j, written = 0L; - int i; - char c; - - rewind (x->fP); - fprintf (x->fP, "MThd"); - j = 6; // length of header data - for (i = 0; i < 4; ++i) - { // msb first - c = (char)((j & 0xFF000000)>>24); - putc(c, x->fP); - j <<= 8; - } - j = 0; // type of file - for (i = 0; i < 2; ++i) - { // msb first - c = (char)((j & 0xFF00)>>8); - putc(c, x->fP); - j <<= 8; - } - j = 1; // number of tracks - for (i = 0; i < 2; ++i) - { // msb first - c = (char)((j & 0xFF00)>>8); - putc(c, x->fP); - j <<= 8; - } - j = x->header_chunk.chunk_division; // ticks per quarter note - for (i = 0; i < 2; ++i) - { // msb first - c = (char)((j & 0xFF00)>>8); - putc(c, x->fP); - j <<= 8; - } - fprintf (x->fP, "MTrk"); - j = x->track_chunk[0].chunk_length; // length of MIDI data -// if (x->verbosity) post ("chunk_length is %lu", j); - for (i = 0; i < 4; ++i) - { // msb first - c = (char)((j & 0xFF000000)>>24); - putc(c, x->fP); -// if (x->verbosity) post ("...%d...", c); - j <<= 8; - } - written = 22L; - return written; + size_t j, written = 0L; + int i; + char c; + + rewind (x->fP); + fprintf (x->fP, "MThd"); + j = 6; /* length of header data */ + for (i = 0; i < 4; ++i) + { /* msb first */ + c = (char)((j & 0xFF000000)>>24); + putc(c, x->fP); + j <<= 8; + } + j = 0; /* type of file */ + for (i = 0; i < 2; ++i) + { /* msb first */ + c = (char)((j & 0xFF00)>>8); + putc(c, x->fP); + j <<= 8; + } + j = 1; /* number of tracks */ + for (i = 0; i < 2; ++i) + { /* msb first */ + c = (char)((j & 0xFF00)>>8); + putc(c, x->fP); + j <<= 8; + } + j = x->header_chunk.chunk_division; /* ticks per quarter note */ + for (i = 0; i < 2; ++i) + { /* msb first */ + c = (char)((j & 0xFF00)>>8); + putc(c, x->fP); + j <<= 8; + } + fprintf (x->fP, "MTrk"); + j = x->track_chunk[0].chunk_length; /* length of MIDI data */ + for (i = 0; i < 4; ++i) + { /* msb first */ + c = (char)((j & 0xFF000000)>>24); + putc(c, x->fP); + j <<= 8; + } + written = 22L; + return written; } void midifile_write(t_midifile *x, t_symbol *path) /* open the file for writing and write the header */ { - midifile_free_file(x); - if (midifile_open_path(x, path->s_name, "wb")) - { - if (x->verbosity) post("midifile: opened %s", x->fPath); - x->state = mfWriting; - x->tmpFP = tmpfile (); // a temporary file for the MIDI data while we don't know how long it is - - strncpy (x->header_chunk.chunk_type, "MThd", 4L);// track header chunk - x->header_chunk.chunk_length = 6L; // 3 ints to follow - x->header_chunk.chunk_format = 0; // single-track file - x->header_chunk.chunk_ntrks = 1; // one track for type 0 file - x->header_chunk.chunk_division = 0; // for now - strncpy (x->track_chunk[0].chunk_type, "MTrk", 4L); - x->track_chunk[0].chunk_length = 0L; // for now - midifile_rewind_tracks(x); - } - else error("midifile: Unable to open %s", path->s_name); + midifile_free_file(x); + if (midifile_open_path(x, path->s_name, "wb")) + { + if (x->verbosity) post("midifile: opened %s", x->fPath); + x->state = mfWriting; + x->tmpFP = tmpfile (); /* a temporary file for the MIDI data while we don't know how long it is */ + + strncpy (x->header_chunk.chunk_type, "MThd", 4L);/* track header chunk */ + x->header_chunk.chunk_length = 6L; /* 3 ints to follow */ + x->header_chunk.chunk_format = 0; /* single-track file */ + x->header_chunk.chunk_ntrks = 1; /* one track for type 0 file */ + x->header_chunk.chunk_division = 0; /* for now */ + strncpy (x->track_chunk[0].chunk_type, "MTrk", 4L); + x->track_chunk[0].chunk_length = 0L; /* for now */ + midifile_rewind_tracks(x); + } + else error("midifile: Unable to open %s", path->s_name); } void midifile_read(t_midifile *x, t_symbol *path) { - midifile_free_file(x); - if (midifile_open_path(x, path->s_name, "rb")) - { - if (x->verbosity) post("midifile: opened %s", x->fPath); - x->state = mfReading; - if (midifle_read_chunks(x) == 0) midifile_free_file(x); - } - else error("midifile: Unable to open %s", path->s_name); + midifile_free_file(x); + if (midifile_open_path(x, path->s_name, "rb")) + { + if (x->verbosity) post("midifile: opened %s", x->fPath); + x->state = mfReading; + if (midifle_read_chunks(x) == 0) midifile_free_file(x); + } + else error("midifile: Unable to open %s", path->s_name); } void midifile_bang(t_midifile *x) /* step forward one tick and process all tracks for that tick */ { - int j, result = 1, ended = 0; - size_t total_time; - - switch (x->state) - { - case mfReading: - if (x->verbosity > 2) post("midifile_bang: total_time %lu", x->total_time); - for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) - { - if (x->track_chunk[j].total_time != NO_MORE_ELEMENTS) - { - while ((total_time = midifile_get_next_track_chunk_delta_time(x, j) + x->track_chunk[j].total_time) == x->total_time) - { - if ((x->track == j) ||(x->track == ALL_TRACKS)) midifile_get_next_track_chunk_data(x, j); - else midifile_skip_next_track_chunk_data(x, j); - } - x->ended = 0; - } - if (x->track_chunk[j].delta_time == NO_MORE_ELEMENTS) ++ended; - } - if ((ended == x->header_chunk.chunk_ntrks)&&(x->ended == 0)) - { // set ended flag, only bang once - if (x->verbosity > 1) post ("ended = %d x->header_chunk.chunk_ntrks = %d", ended, x->header_chunk.chunk_ntrks); - outlet_bang(x->bang_outlet); - ++x->ended; - } - // fall through into mfWriting - case mfWriting: - ++x->total_time; - outlet_float(x->total_time_outlet, x->total_time); - break; - default: - break;// don't change time when no files are open - } + int j, result = 1, ended = 0; + size_t total_time; + + switch (x->state) + { + case mfReading: + if (x->verbosity > 2) post("midifile_bang: total_time %lu", x->total_time); + for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) + { + if (x->track_chunk[j].total_time != NO_MORE_ELEMENTS) + { + while + ( + ( + total_time = midifile_get_next_track_chunk_delta_time(x, j) + + x->track_chunk[j].total_time + ) + == x->total_time + ) + { + if ((x->track == j) ||(x->track == ALL_TRACKS)) midifile_get_next_track_chunk_data(x, j); + else midifile_skip_next_track_chunk_data(x, j); + } + x->ended = 0; + } + if (x->track_chunk[j].delta_time == NO_MORE_ELEMENTS) ++ended; + } + if ((ended == x->header_chunk.chunk_ntrks)&&(x->ended == 0)) + { /* set ended flag, only bang once */ + if (x->verbosity > 1) + post ("ended = %d x->header_chunk.chunk_ntrks = %d", ended, x->header_chunk.chunk_ntrks); + outlet_bang(x->bang_outlet); + ++x->ended; + } + /* fall through into mfWriting */ + case mfWriting: + ++x->total_time; + outlet_float(x->total_time_outlet, x->total_time); + break; + default: + break;/* don't change time when no files are open */ + } } /* The arguments of the ``list''-method @@ -415,383 +416,379 @@ void midifile_bang(t_midifile *x) void midifile_list(t_midifile *x, t_symbol *s, int argc, t_atom *argv) /* add a list containing time and midi packet to the temporary file in MIDI file format */ { - int i, j, k, m, dt_written = 0; - size_t len, written = 0L; - static int warnings = 0; - - if (! x->state == mfWriting) return;// list only works for writing - if (x->tmpFP == NULL) - { - if (0 == warnings++) error ("midifile: no file is open for writing"); - return; - } - for (i = 0; i < argc; ++i) - { - switch (argv[i].a_type) - { - case A_FLOAT: - { - j = atom_getint(&argv[i]); - if (x->verbosity > 2) post ("midifile_list. j[%d] = 0x%lX", i, j); - if (j <= 0x100) - { - if (!dt_written) - { // deltatime - x->track_chunk[0].delta_time = x->total_time - x->track_chunk[0].total_time; - x->track_chunk[0].total_time = x->total_time; - written = midifile_write_variable_length_value(x->tmpFP, x->track_chunk[0].delta_time); - dt_written = 1; - } - if (j == x->track_chunk[0].running_status) break;// don't save redundant status byte - if (j >= 0x80 && j <= 0xEF)x->track_chunk[0].running_status = j;// new running status - else if (j >= 0xF0 && j <= 0xF7) - { - x->track_chunk[0].running_status = 0;// clear running status - if (j == 0xF0) - { // system exclusive: - // find length - for (k = i+1, len = 0L; k < argc; ++k, ++len) - { - if (argv[k].a_type != A_FLOAT) - { - error ("midifile: sysex list must be all floats"); - x->track_chunk[0].chunk_length += written; - return; - } - m = atom_getint(&argv[k]); - if (m & 0x80) break;// take any non-data as end of exclusive - } - if (m != 0xF7) - { - error ("midifile: sysex list terminator is 0x%d", m); - x->track_chunk[0].chunk_length += written; - return; - } - ++len; - if (x->verbosity) post ("midifile: sysex length %lu. j = 0x%X", len, j); - putc (j, x->tmpFP); - ++written; - // write length as variable length - written += midifile_write_variable_length_value (x->tmpFP, len); - // write the rest of the sysex message - for (k = i+1; j != 0xF7; ++k) - { - j = atom_getint(&argv[k]); - putc (j, x->tmpFP); - ++written; - } - x->track_chunk[0].chunk_length += written; - return; - } - } - if (x->verbosity > 1) post ("midifile: j = 0x%X", j); - putc (j, x->tmpFP); - ++written; - } - break; - } - } - } - x->track_chunk[0].chunk_length += written; -// if (x->verbosity) post ("midifile_list"); + int i, j, k, m, dt_written = 0; + size_t len, written = 0L; + static int warnings = 0; + + if (! x->state == mfWriting) return;/* list only works for writing */ + if (x->tmpFP == NULL) + { + if (0 == warnings++) error ("midifile: no file is open for writing"); + return; + } + for (i = 0; i < argc; ++i) + { + if (A_FLOAT == argv[i].a_type) + { + j = atom_getint(&argv[i]); + if (x->verbosity > 2) post ("midifile_list. j[%d] = 0x%lX", i, j); + if (j <= 0x100) + { + if (!dt_written) + { /* deltatime */ + x->track_chunk[0].delta_time = x->total_time - x->track_chunk[0].total_time; + x->track_chunk[0].total_time = x->total_time; + written = midifile_write_variable_length_value(x->tmpFP, x->track_chunk[0].delta_time); + dt_written = 1; + } + if (j == x->track_chunk[0].running_status) break;/* don't save redundant status byte */ + if (j >= 0x80 && j <= 0xEF)x->track_chunk[0].running_status = j;/* new running status */ + else if (j >= 0xF0 && j <= 0xF7) + { + x->track_chunk[0].running_status = 0;/* clear running status */ + if (j == 0xF0) + { /* system exclusive: */ + /* find length */ + for (k = i+1, len = 0L; k < argc; ++k, ++len) + { + if (argv[k].a_type != A_FLOAT) + { + error ("midifile: sysex list must be all floats"); + x->track_chunk[0].chunk_length += written; + return; + } + m = atom_getint(&argv[k]); + if (m & 0x80) break;/* take any non-data as end of exclusive */ + } + if (m != 0xF7) + { + error ("midifile: sysex list terminator is 0x%d", m); + x->track_chunk[0].chunk_length += written; + return; + } + ++len; + if (x->verbosity) post ("midifile: sysex length %lu. j = 0x%X", len, j); + putc (j, x->tmpFP); + ++written; + /* write length as variable length */ + written += midifile_write_variable_length_value (x->tmpFP, len); + /* write the rest of the sysex message */ + for (k = i+1; j != 0xF7; ++k) + { + j = atom_getint(&argv[k]); + putc (j, x->tmpFP); + ++written; + } + x->track_chunk[0].chunk_length += written; + return; + } + } + if (x->verbosity > 1) post ("midifile: j = 0x%X", j); + putc (j, x->tmpFP); + ++written; + } + } + } + x->track_chunk[0].chunk_length += written; } size_t midifile_write_end_of_track(t_midifile *x, size_t end_time) /* write End of Track event to x->tmpFP */ { - size_t written = 0L; - - x->track_chunk[0].delta_time = end_time - x->track_chunk[0].total_time; - x->track_chunk[0].total_time = x->total_time; - written = midifile_write_variable_length_value (x->tmpFP, x->track_chunk[0].delta_time); - putc (0xFF, x->tmpFP); - putc (0x2F, x->tmpFP); - putc (0x00, x->tmpFP); - written += 3L; - x->track_chunk[0].chunk_length += written; - return written; + size_t written = 0L; + + x->track_chunk[0].delta_time = end_time - x->track_chunk[0].total_time; + x->track_chunk[0].total_time = x->total_time; + written = midifile_write_variable_length_value (x->tmpFP, x->track_chunk[0].delta_time); + putc (0xFF, x->tmpFP); + putc (0x2F, x->tmpFP); + putc (0x00, x->tmpFP); + written += 3L; + x->track_chunk[0].chunk_length += written; + return written; } void midifile_float(t_midifile *x, t_float ticks) /* go to a total time of cue_time */ { - size_t cTime = (size_t)ticks; - size_t total_time; - int j, result = 1, ended = 0; - - switch (x->state) - { - case mfReading: // cue to ticks - //if (x->verbosity) post ("midifile_float %d", cTime); - midifile_rewind_tracks(x); - for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) - { - if (x->track_chunk[j].total_time != NO_MORE_ELEMENTS) - { - while ((total_time = midifile_get_next_track_chunk_delta_time(x, j) + x->track_chunk[j].total_time) < cTime) - { - midifile_skip_next_track_chunk_data(x, j); - } - } - if (x->track_chunk[j].delta_time == NO_MORE_ELEMENTS) ++ended; - } - x->total_time = cTime; - outlet_float(x->total_time_outlet, x->total_time); - if (ended == x->header_chunk.chunk_ntrks) - { - if (x->verbosity) post ("midifile: ended = %d x->header_chunk.chunk_ntrks = %d", ended, x->header_chunk.chunk_ntrks); - outlet_bang(x->bang_outlet); - } - break; - case mfWriting: // add ticks to current time - x->total_time += cTime; - outlet_float(x->total_time_outlet, x->total_time); - break; - case mfReset: // do nothing - break; - } - + size_t cTime = (size_t)ticks; + size_t total_time; + int j, result = 1, ended = 0; + + switch (x->state) + { + case mfReading: /* cue to ticks */ + midifile_rewind_tracks(x); + for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) + { + if (x->track_chunk[j].total_time != NO_MORE_ELEMENTS) + { + while + ( + ( + total_time = midifile_get_next_track_chunk_delta_time(x, j) + + x->track_chunk[j].total_time + ) + < cTime + ) + midifile_skip_next_track_chunk_data(x, j); + } + if (x->track_chunk[j].delta_time == NO_MORE_ELEMENTS) ++ended; + } + x->total_time = cTime; + outlet_float(x->total_time_outlet, x->total_time); + if (ended == x->header_chunk.chunk_ntrks) + { + if (x->verbosity) + post ("midifile: ended = %d x->header_chunk.chunk_ntrks = %d", ended, x->header_chunk.chunk_ntrks); + outlet_bang(x->bang_outlet); + } + break; + case mfWriting: /* add ticks to current time */ + x->total_time += cTime; + outlet_float(x->total_time_outlet, x->total_time); + break; + case mfReset: /* do nothing */ + break; + } } int midifle_read_chunks(t_midifile *x) { - int j, result; -// size_t delta_time; - - result = midifile_read_header_chunk(x); - midifile_rewind_tracks(x); - for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) - midifile_read_track_chunk(x, j); - return result; + int j, result; + + result = midifile_read_header_chunk(x); + midifile_rewind_tracks(x); + for (j = 0; ((j < x->header_chunk.chunk_ntrks)&&(result != 0)); ++j) + midifile_read_track_chunk(x, j); + return result; } int midifile_read_header_chunk(t_midifile *x) { - char *cP = x->header_chunk.chunk_type; - char *sP; - char buf[4]; - size_t n; - int div, smpte, ticks; - - if (x->fP == NULL) - { - error("midifile: no open file"); - return 0;// no open file - } - rewind(x->fP); - x->offset = 0L; - n = fread(cP, 1L, 4L, x->fP); - x->offset += n; - if (n != 4L) - { - error("midifile: read %lu instead of 4", n); - return 0; - } - if (x->verbosity) post("midifile: Header chunk type: %c%c%c%c", cP[0], cP[1], cP[2], cP[3]); - if (!(cP[0] == 'M' && cP[1] == 'T' && cP[2] == 'h' && cP[3] == 'd')) - { - error ("midifile: bad file format: bad header chunk type"); - return 0; - } - cP = buf; - n = fread(cP, 1L, 4L, x->fP); - x->offset += n; - if (n != 4L) - { - error("midifile: read %lu instead of 4", n); - return 0; - } - x->header_chunk.chunk_length = midifile_get_multibyte_4(cP); - if (x->verbosity) post("midifile: Header chunk length: %lu", x->header_chunk.chunk_length); - if (x->header_chunk.chunk_length != 6L) - { - error ("midifile: bad file format: bad header chunk length"); - return 0; - } - n = fread(cP, 1L, 2L, x->fP); - x->offset += n; - if (n != 2L) - { - error("midifile: read %lu instead of 2", n); - return 0; - } - x->header_chunk.chunk_format = midifile_get_multibyte_2(cP); - switch (x->header_chunk.chunk_format) - { - case 0: - sP = "Single multichannel track"; - break; - case 1: - sP = "One or more simultaneous tracks"; - break; - case 2: - sP = "One or more sequentially independent single tracks"; - break; - default: - sP = "Unknown format"; - } - if (x->verbosity) post("midifile: Header chunk format: %d (%s)", x->header_chunk.chunk_format, sP); - - n = fread(cP, 1L, 2L, x->fP); - x->offset += n; - if (n != 2L) - { - error("midifile: read %lu instead of 2", n); - return 0; - } - x->header_chunk.chunk_ntrks = midifile_get_multibyte_2(cP); - if (x->verbosity) post("midifile: Header chunk ntrks: %d", x->header_chunk.chunk_ntrks); - if (x->header_chunk.chunk_ntrks > MAX_TRACKS) - { - error ("midifile: Header chunk ntrks (%d) exceeds midifile MAX_TRACKS, set to %d", x->header_chunk.chunk_ntrks, MAX_TRACKS); - x->header_chunk.chunk_ntrks = MAX_TRACKS; - } - - n = fread(cP, 1L, 2L, x->fP); - x->offset += n; - if (n != 2L) - { - error("midifile: read %lu instead of 2", n); - return 0; - } - x->header_chunk.chunk_division = midifile_get_multibyte_2(cP); - div = x->header_chunk.chunk_division; - if(div & 0x8000) - { - smpte = (-(div>>8)) & 0x0FF; - ticks = div & 0x0FF; - if (x->verbosity) post("midifile: Header chunk division: 0x%X: %d frames per second, %d ticks per frame", div, smpte, ticks); - } - else - { - if (x->verbosity) post("midifile: Header chunk division: 0x%X: %d ticks per quarter note", div, div); - } - return 1; + char *cP = x->header_chunk.chunk_type; + char *sP; + char buf[4]; + size_t n; + int div, smpte, ticks; + + if (x->fP == NULL) + { + error("midifile: no open file"); + return 0;/* no open file */ + } + rewind(x->fP); + x->offset = 0L; + n = fread(cP, 1L, 4L, x->fP); + x->offset += n; + if (n != 4L) + { + error("midifile: read %lu instead of 4", n); + return 0; + } + if (x->verbosity) post("midifile: Header chunk type: %c%c%c%c", cP[0], cP[1], cP[2], cP[3]); + if (!(cP[0] == 'M' && cP[1] == 'T' && cP[2] == 'h' && cP[3] == 'd')) + { + error ("midifile: bad file format: bad header chunk type"); + return 0; + } + cP = buf; + n = fread(cP, 1L, 4L, x->fP); + x->offset += n; + if (n != 4L) + { + error("midifile: read %lu instead of 4", n); + return 0; + } + x->header_chunk.chunk_length = midifile_get_multibyte_4(cP); + if (x->verbosity) post("midifile: Header chunk length: %lu", x->header_chunk.chunk_length); + if (x->header_chunk.chunk_length != 6L) + { + error ("midifile: bad file format: bad header chunk length"); + return 0; + } + n = fread(cP, 1L, 2L, x->fP); + x->offset += n; + if (n != 2L) + { + error("midifile: read %lu instead of 2", n); + return 0; + } + x->header_chunk.chunk_format = midifile_get_multibyte_2(cP); + switch (x->header_chunk.chunk_format) + { + case 0: + sP = "Single multichannel track"; + break; + case 1: + sP = "One or more simultaneous tracks"; + break; + case 2: + sP = "One or more sequentially independent single tracks"; + break; + default: + sP = "Unknown format"; + } + if (x->verbosity) post("midifile: Header chunk format: %d (%s)", x->header_chunk.chunk_format, sP); + + n = fread(cP, 1L, 2L, x->fP); + x->offset += n; + if (n != 2L) + { + error("midifile: read %lu instead of 2", n); + return 0; + } + x->header_chunk.chunk_ntrks = midifile_get_multibyte_2(cP); + if (x->verbosity) post("midifile: Header chunk ntrks: %d", x->header_chunk.chunk_ntrks); + if (x->header_chunk.chunk_ntrks > MAX_TRACKS) + { + error ("midifile: Header chunk ntrks (%d) exceeds midifile MAX_TRACKS, set to %d", + x->header_chunk.chunk_ntrks, MAX_TRACKS); + x->header_chunk.chunk_ntrks = MAX_TRACKS; + } + n = fread(cP, 1L, 2L, x->fP); + x->offset += n; + if (n != 2L) + { + error("midifile: read %lu instead of 2", n); + return 0; + } + x->header_chunk.chunk_division = midifile_get_multibyte_2(cP); + div = x->header_chunk.chunk_division; + if(div & 0x8000) + { + smpte = (-(div>>8)) & 0x0FF; + ticks = div & 0x0FF; + if (x->verbosity) + post("midifile: Header chunk division: 0x%X: %d frames per second, %d ticks per frame", div, smpte, ticks); + } + else if (x->verbosity) post("midifile: Header chunk division: 0x%X: %d ticks per quarter note", div, div); + return 1; } int midifile_read_track_chunk(t_midifile *x, int index) /* read the data part of a track chunk into track_data */ /* after allocating the space for it */ { - char *cP = x->track_chunk[index].chunk_type; - char buf[4]; - char type[5]; - size_t n, len; - - if (x->fP == NULL) - { - error("midifile: no open file"); - return 0;// no open file - } - n = fread(cP, 1L, 4L, x->fP); - x->offset += n; - if (n != 4L) - { - error("midifile: read %lu instead of 4", n); - return 0; - } - if (!(cP[0] == 'M' && cP[1] == 'T' && cP[2] == 'r' && cP[3] == 'k')) - { - error ("midifile: bad file format: bad track chunk type"); - return 0; - } - type[0] = cP[0]; - type[1] = cP[1]; - type[2] = cP[2]; - type[3] = cP[3]; - type[4] = '\0'; - cP = buf; - n = fread(cP, 1L, 4L, x->fP); - x->offset += n; - if (n != 4L) - { - error("midifile: read %lu instead of 4", n); - return 0; - } - len = midifile_get_multibyte_4(cP); - x->track_chunk[index].chunk_length = len; - if (x->verbosity) post("midifile: Track chunk %d type: %s, length %lu", index, type, len); - if ((cP = getbytes(len)) == NULL) - { - error ("midifile: Unable to allocate %lu bytes for track data", len); - return 0; - } - x->track_chunk[index].track_data = cP; - n = fread(cP, 1L, len, x->fP); - - return 1; + char *cP = x->track_chunk[index].chunk_type; + char buf[4]; + char type[5]; + size_t n, len; + + if (x->fP == NULL) + { + error("midifile: no open file"); + return 0;/* no open file */ + } + n = fread(cP, 1L, 4L, x->fP); + x->offset += n; + if (n != 4L) + { + error("midifile: read %lu instead of 4", n); + return 0; + } + if (!(cP[0] == 'M' && cP[1] == 'T' && cP[2] == 'r' && cP[3] == 'k')) + { + error ("midifile: bad file format: bad track chunk type"); + return 0; + } + type[0] = cP[0]; + type[1] = cP[1]; + type[2] = cP[2]; + type[3] = cP[3]; + type[4] = '\0'; + cP = buf; + n = fread(cP, 1L, 4L, x->fP); + x->offset += n; + if (n != 4L) + { + error("midifile: read %lu instead of 4", n); + return 0; + } + len = midifile_get_multibyte_4(cP); + x->track_chunk[index].chunk_length = len; + if (x->verbosity) post("midifile: Track chunk %d type: %s, length %lu", index, type, len); + if ((cP = getbytes(len)) == NULL) + { + error ("midifile: Unable to allocate %lu bytes for track data", len); + return 0; + } + x->track_chunk[index].track_data = cP; + n = fread(cP, 1L, len, x->fP); + + return 1; } unsigned short midifile_combine_bytes(unsigned char data1, unsigned char data2) /* make a short from two 7bit MIDI data bytes */ { /* - unsigned short value = (unsigned short)data2; - value <<= 7; - value |= (unsigned short)data1; - return value; + unsigned short value = (unsigned short)data2; + value <<= 7; + value |= (unsigned short)data1; + return value; */ - return ((((unsigned short)data2)<< 7) | ((unsigned short)data1)); + return ((((unsigned short)data2)<< 7) | ((unsigned short)data1)); } unsigned long midifile_get_multibyte_4(char*n) /* make a long from 4 consecutive bytes in big-endian format */ { - unsigned long a, b, c, d, e; - a = (*(unsigned long *)(&n[0])) & 0x0FF; - b = (*(unsigned long *)(&n[1])) & 0x0FF; - c = (*(unsigned long *)(&n[2])) & 0x0FF; - d = (*(unsigned long *)(&n[3])) &0x0FF; - e = (a<<24) + (b<<16) + (c<<8) + d; - return e; + unsigned long a, b, c, d, e; + a = (*(unsigned long *)(&n[0])) & 0x0FF; + b = (*(unsigned long *)(&n[1])) & 0x0FF; + c = (*(unsigned long *)(&n[2])) & 0x0FF; + d = (*(unsigned long *)(&n[3])) &0x0FF; + e = (a<<24) + (b<<16) + (c<<8) + d; + return e; } unsigned long midifile_get_multibyte_3(char*n) /* make a long from 3 consecutive bytes in big-endian format */ { - unsigned long a, b, c, d; - a = (*(unsigned long *)(&n[0])) & 0x0FF; - b = (*(unsigned long *)(&n[1])) & 0x0FF; - c = (*(unsigned long *)(&n[2])) & 0x0FF; - d = (a<<16) + (b<<8) + c; - return d; + unsigned long a, b, c, d; + a = (*(unsigned long *)(&n[0])) & 0x0FF; + b = (*(unsigned long *)(&n[1])) & 0x0FF; + c = (*(unsigned long *)(&n[2])) & 0x0FF; + d = (a<<16) + (b<<8) + c; + return d; } unsigned short midifile_get_multibyte_2(char*n) /* make a short from 2 consecutive bytes in big-endian format */ { - unsigned short a, b, c; - a = (*(unsigned long *)(&n[0])) & 0x0FF; - b = (*(unsigned long *)(&n[1])) & 0x0FF; - c = (a<<8) + b; - return c; + unsigned short a, b, c; + a = (*(unsigned long *)(&n[0])) & 0x0FF; + b = (*(unsigned long *)(&n[1])) & 0x0FF; + c = (a<<8) + b; + return c; } int midifile_write_variable_length_value (FILE *fP, size_t value) /* return number of characters written to fP */ { - size_t buffer; - int i; - char c; - - buffer = value & 0x07F; - while ((value >>= 7) > 0) - { - buffer <<= 8; - buffer |= 0x80; - buffer += (value & 0x07F); - } - i = 0; - while (1) - { - c = (char)(buffer & (0x0FF)); - putc(c, fP); - ++i; - if (buffer & 0x80) buffer >>= 8; - else break; - } - return i; + size_t buffer; + int i; + char c; + + buffer = value & 0x07F; + while ((value >>= 7) > 0) + { + buffer <<= 8; + buffer |= 0x80; + buffer += (value & 0x07F); + } + i = 0; + while (1) + { + c = (char)(buffer & (0x0FF)); + putc(c, fP); + ++i; + if (buffer & 0x80) buffer >>= 8; + else break; + } + return i; } char *midifile_read_var_len (char *cP, size_t *delta) @@ -800,645 +797,650 @@ char *midifile_read_var_len (char *cP, size_t *delta) /* set delta to deltatime */ /* return pointer to following data */ - unsigned long value; - char c; - - if (((value = *(cP++))) & 0x80) - { - value &= 0x7f; - do - { - value = (value << 7) + ((c = *(cP++)) & 0x7f); - } while (c & 0x80); - } - *delta = value; - return cP; + unsigned long value; + char c; + + if (((value = *(cP++))) & 0x80) + { + value &= 0x7f; + do + { + value = (value << 7) + ((c = *(cP++)) & 0x7f); + } while (c & 0x80); + } + *delta = value; + return cP; } void midifile_verbosity(t_midifile *x, t_floatarg verbosity) /* set verbosity of console output */ { - x->verbosity = verbosity; + x->verbosity = verbosity; } void midifile_single_track(t_midifile *x, t_floatarg track) /* play only this track or all tracks if out of range */ { - if(x->state != mfReading) return; // only if we're reading - if ((track < 0) || (track >= x->header_chunk.chunk_ntrks)) - // anything out of range will be interpreted as all tracks - x->track = ALL_TRACKS; - else x->track = track; + if(x->state != mfReading) return; /* only if we're reading */ + if ((track < 0) || (track >= x->header_chunk.chunk_ntrks)) + /* anything out of range will be interpreted as all tracks */ + x->track = ALL_TRACKS; + else x->track = track; } void midifile_dump(t_midifile *x, t_floatarg track) { - int index = (int)track; - - if(x->state != mfReading) return; // only if we're reading - if ((index < x->header_chunk.chunk_ntrks) && (index >= 0)) - midifile_dump_track_chunk_data(x, index); - else// anything out of range will be interpreted as all tracks - for (index = 0; index < x->header_chunk.chunk_ntrks; ++index) - midifile_dump_track_chunk_data(x, index); + int index = (int)track; + + if(x->state != mfReading) return; /* only if we're reading */ + if ((index < x->header_chunk.chunk_ntrks) && (index >= 0)) + midifile_dump_track_chunk_data(x, index); + else /* anything out of range will be interpreted as all tracks */ + for (index = 0; index < x->header_chunk.chunk_ntrks; ++index) + midifile_dump_track_chunk_data(x, index); } void midifile_rewind (t_midifile *x) { - if(x->state != mfReading) return; // only if we're reading - midifile_rewind_tracks(x); + if(x->state != mfReading) return; /* only if we're reading */ + midifile_rewind_tracks(x); } void midifile_rewind_tracks(t_midifile *x) /* For all tracks, point to start of track_data */ { - int i; - for (i = 0; i < x->header_chunk.chunk_ntrks; ++i) - { - x->track_chunk[i].delta_time = 0L; - x->track_chunk[i].track_index = 0L; - x->track_chunk[i].total_time = 0L; - x->track_chunk[i].running_status = 0; - } - x->total_time = 0L; - x->ended = 0L; - outlet_float(x->total_time_outlet, x->total_time); + int i; + for (i = 0; i < x->header_chunk.chunk_ntrks; ++i) + { + x->track_chunk[i].delta_time = 0L; + x->track_chunk[i].track_index = 0L; + x->track_chunk[i].total_time = 0L; + x->track_chunk[i].running_status = 0; + } + x->total_time = 0L; + x->ended = 0L; + outlet_float(x->total_time_outlet, x->total_time); } size_t midifile_get_next_track_chunk_delta_time(t_midifile *x, int index) /* return the delta_time of the next event in track[index] */ { - unsigned char *cP, *last_cP; - size_t delta_time; + unsigned char *cP, *last_cP; + size_t delta_time; - cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; - last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; + cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; + last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; - delta_time = NO_MORE_ELEMENTS; - if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) cP = midifile_read_var_len(cP, &delta_time); -// if (x->verbosity) post("delta time for track[%d]: %lu", index, delta_time); - return delta_time; + delta_time = NO_MORE_ELEMENTS; + if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) + cP = midifile_read_var_len(cP, &delta_time); + return delta_time; } void midifile_output_long_list (t_outlet *outlet, unsigned char *cP, size_t len, unsigned char first_byte) -{ // ouput a long MIDI message as a list of floats - // first_byte is followed by len bytes at cP - size_t slen; - unsigned int si; - t_atom *slist; - - slen = (len+1L)*sizeof(t_atom); - slist = getbytes (slen); - if (slist == NULL) - { - error ("midifile: no memory for long list"); - return; - } - slist[0].a_type = A_FLOAT; - slist[0].a_w.w_float = 0xF0; - for (si = 0; si < len; ++si) - { - slist[si+1].a_type = A_FLOAT; - slist[si+1].a_w.w_float = cP[si]; - } - outlet_list(outlet, &s_list, len+1L, slist); - freebytes(slist, slen); +{ /* output a long MIDI message as a list of floats */ + /* first_byte is followed by len bytes at cP */ + size_t slen; + unsigned int si; + t_atom *slist; + + slen = (len+1L)*sizeof(t_atom); + slist = getbytes (slen); + if (slist == NULL) + { + error ("midifile: no memory for long list"); + return; + } + slist[0].a_type = A_FLOAT; + slist[0].a_w.w_float = 0xF0; + for (si = 0; si < len; ++si) + { + slist[si+1].a_type = A_FLOAT; + slist[si+1].a_w.w_float = cP[si]; + } + outlet_list(outlet, &s_list, len+1L, slist); + freebytes(slist, slen); } void midifile_dump_track_chunk_data(t_midifile *x, int index) /* parse entire track chunk and output it to the main window */ { - unsigned char *cP, *last_cP, *str; - size_t total_time, delta_time, time_sig, len; - unsigned char status, running_status, c, d, nn, dd, cc, bb, mi, mcp, ch; - char sf; - unsigned short sn; - unsigned char tt[3]; - char *msgPtr; - char msg[256]; - - cP = x->track_chunk[index].track_data; - last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; - total_time = 0L; - - post("midifile: Parsing track[%d]...", index); - while ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) - { - msgPtr = msg; - cP = midifile_read_var_len(cP, &delta_time); - status = *cP++; - total_time += delta_time; - msgPtr += sprintf (msgPtr, "tick %lu delta %lu status %02X ", total_time, delta_time, status); - if ((status & 0xF0) == 0xF0) - { - switch (status) - { - case 0xF0: - case 0xF7: - cP = midifile_read_var_len(cP, &len);// not a time but the same variable length format - msgPtr += sprintf(msgPtr, "Sysex: %02X length %lu ", status, len); - cP += len; - break; - case 0xF3: // song select - c = *cP++; - msgPtr += sprintf(msgPtr, "Song Select: %d ", c); - break; - case 0xF2: // song position - c = *cP++; - d = *cP++; - msgPtr += sprintf(msgPtr, "Song Position %d ", midifile_combine_bytes(c, d)); - break; - case 0xF1: // quarter frame - msgPtr += sprintf(msgPtr, "MIDI Quarter Frame"); - break; - case 0xF6: // tune request - msgPtr += sprintf(msgPtr, "MIDI Tune Request"); - break; - case 0xF8: // MIDI clock - msgPtr += sprintf(msgPtr, "MIDI Clock"); - break; - case 0xF9: // MIDI tick - msgPtr += sprintf(msgPtr, "MIDI Tick"); - break; - case 0xFA: // MIDI start - msgPtr += sprintf(msgPtr, "MIDI Start"); - break; - case 0xFB: // MIDI continue - msgPtr += sprintf(msgPtr, "MIDI Continue"); - break; - case 0xFC: // MIDI stop - msgPtr += sprintf(msgPtr, "MIDI Stop"); - break; - case 0xFE: // active sense - msgPtr += sprintf(msgPtr, "MIDI Active Sense"); - break; - case 0xFF: - c = *cP++; - cP = midifile_read_var_len(cP, &len);// not a time but the same variable length format - msgPtr += sprintf(msgPtr, "Meta 0x%02X length %lu \n", c, len); - switch (c) - { - case 0x58: - nn = *cP++; - dd = *cP++; - dd = 1<<(dd); - cc = *cP++; - bb = *cP++; - msgPtr += sprintf(msgPtr, "Time Signature %d/%d %d clocks per tick, %d 32nd notes per quarter note", nn, dd, cc, bb); - break; - case 0x59: - sf = *(signed char*)cP++; - mi = *cP++; - msgPtr += sprintf(msgPtr, "Key Signature: %d %s, %s", sf, (sf<0)?"flats":"sharps", (mi)?"minor":"major"); - break; - case 0x51: - tt[0] = *cP++; - tt[1] = *cP++; - tt[2] = *cP++; - time_sig = midifile_get_multibyte_3(tt); - msgPtr += sprintf(msgPtr, "%lu microseconds per MIDI quarter-note", time_sig); - break; - case 0x2F: - msgPtr += sprintf(msgPtr, "========End of Track %d==========", index); - cP += len; - break; - case 0x21: - tt[0] = *cP++; - msgPtr += sprintf(msgPtr, "MIDI port or cable number (unofficial): %d", tt[0]); - break; - case 0x20: - mcp = *cP++; - msgPtr += sprintf(msgPtr, "MIDI Channel Prefix: %d", mcp); - break; - case 0x06: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Marker %s", str); - cP[len] = c; - cP += len; - break; - case 0x05: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Lyric %s", str); - cP[len] = c; - cP += len; - break; - case 0x04: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Instrument Name %s", str); - cP[len] = c; - cP += len; - break; - case 0x03: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Sequence/Track Name %s", str); - cP[len] = c; - cP += len; - break; - case 0x02: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Copyright Notice %s", str); - cP[len] = c; - cP += len; - break; - case 0x01: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - msgPtr += sprintf(msgPtr, "Text %s", str); - cP[len] = c; - cP += len; - break; - case 0x00: - tt[0] = *cP++; - tt[1] = *cP++; - sn = midifile_get_multibyte_2(tt); - msgPtr += sprintf(msgPtr, "Sequence Number %d", sn); - break; - default: - msgPtr += sprintf(msgPtr, "Unknown: 0x%02X", c); - cP += len; - break; - } - break; - default: // 0xF4, 0xF5, 0xF9, 0xFD are not defined - msgPtr += sprintf(msgPtr, "Undefined: 0x%02X", status); - break; - } - } - else - { - if (status & 0x80) - { - running_status = status; - c = *cP++; - } - else - { - c = status; - status = running_status; - } - ch = (status & 0x0F) + 1; // MIDI channel number - switch (status & 0xF0) - { - case 0x80: - d = *cP++; // 2 data bytes - msgPtr += sprintf(msgPtr, "MIDI 0x%02X %02X %02X : channel %d Note %d Off velocity %d", status, c, d, ch, c, d); - break; - case 0x90: - d = *cP++; // 2 data bytes - if (d == 0) - msgPtr += sprintf(msgPtr,"MIDI 0x%02X %02X %02X : channel %d Note %d Off", status, c, d, ch, c); - else - msgPtr += sprintf(msgPtr,"MIDI 0x%02X %02X %02X : channel %d Note %d On velocity %d", status, c, d, ch, c, d); - break; - case 0xA0: - d = *cP++; // 2 data bytes - msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X %02X : channel %d Note %d Aftertouch %d", status, c, d, ch, c, d); - break; - case 0xB0: - d = *cP++; // 2 data bytes - msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X %02X : channel %d Controller %d: %d", status, c, d, ch, c, d); - break; - case 0xC0: // 1 data byte - msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X: channel %d Program Change: %d", status, c, ch, c); - break; - case 0xD0: // 1 data byte - msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X: channel %d Channel Pressure: %d", status, c, ch, c); - break; - case 0xE0: // 2 data bytes - d = *cP++; // 2 data bytes - msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X %02X : channel %d Pitch Wheel %d", status, c, d, ch, midifile_combine_bytes(c, d)); - break; - } - } - post("midifile: %s", msg); - } + unsigned char *cP, *last_cP, *str; + size_t total_time, delta_time, time_sig, len; + unsigned char status, running_status, c, d, nn, dd, cc, bb, mi, mcp, ch; + char sf; + unsigned short sn; + unsigned char tt[3]; + char *msgPtr; + char msg[256]; + + cP = x->track_chunk[index].track_data; + last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; + total_time = 0L; + + post("midifile: Parsing track[%d]...", index); + while ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) + { + msgPtr = msg; + cP = midifile_read_var_len(cP, &delta_time); + status = *cP++; + total_time += delta_time; + msgPtr += sprintf (msgPtr, "tick %lu delta %lu status %02X ", total_time, delta_time, status); + if ((status & 0xF0) == 0xF0) + { + switch (status) + { + case 0xF0: + case 0xF7: + cP = midifile_read_var_len(cP, &len);/* not a time but the same variable length format */ + msgPtr += sprintf(msgPtr, "Sysex: %02X length %lu ", status, len); + cP += len; + break; + case 0xF3: /* song select */ + c = *cP++; + msgPtr += sprintf(msgPtr, "Song Select: %d ", c); + break; + case 0xF2: /* song position */ + c = *cP++; + d = *cP++; + msgPtr += sprintf(msgPtr, "Song Position %d ", midifile_combine_bytes(c, d)); + break; + case 0xF1: /* quarter frame */ + msgPtr += sprintf(msgPtr, "MIDI Quarter Frame"); + break; + case 0xF6: /* tune request */ + msgPtr += sprintf(msgPtr, "MIDI Tune Request"); + break; + case 0xF8: /* MIDI clock */ + msgPtr += sprintf(msgPtr, "MIDI Clock"); + break; + case 0xF9: /* MIDI tick */ + msgPtr += sprintf(msgPtr, "MIDI Tick"); + break; + case 0xFA: /* MIDI start */ + msgPtr += sprintf(msgPtr, "MIDI Start"); + break; + case 0xFB: /* MIDI continue */ + msgPtr += sprintf(msgPtr, "MIDI Continue"); + break; + case 0xFC: /* MIDI stop */ + msgPtr += sprintf(msgPtr, "MIDI Stop"); + break; + case 0xFE: /* active sense */ + msgPtr += sprintf(msgPtr, "MIDI Active Sense"); + break; + case 0xFF: + c = *cP++; + cP = midifile_read_var_len(cP, &len);/* not a time but the same variable length format */ + msgPtr += sprintf(msgPtr, "Meta 0x%02X length %lu \n", c, len); + switch (c) + { + case 0x58: + nn = *cP++; + dd = *cP++; + dd = 1<<(dd); + cc = *cP++; + bb = *cP++; + msgPtr += sprintf( + msgPtr, "Time Signature %d/%d %d clocks per tick, %d 32nd notes per quarter note", + nn, dd, cc, bb); + break; + case 0x59: + sf = *(signed char*)cP++; + mi = *cP++; + msgPtr += sprintf( + msgPtr, "Key Signature: %d %s, %s", + sf, (sf<0)?"flats":"sharps", (mi)?"minor":"major"); + break; + case 0x51: + tt[0] = *cP++; + tt[1] = *cP++; + tt[2] = *cP++; + time_sig = midifile_get_multibyte_3(tt); + msgPtr += sprintf(msgPtr, "%lu microseconds per MIDI quarter-note", time_sig); + break; + case 0x2F: + msgPtr += sprintf(msgPtr, "========End of Track %d==========", index); + cP += len; + break; + case 0x21: + tt[0] = *cP++; + msgPtr += sprintf(msgPtr, "MIDI port or cable number (unofficial): %d", tt[0]); + break; + case 0x20: + mcp = *cP++; + msgPtr += sprintf(msgPtr, "MIDI Channel Prefix: %d", mcp); + break; + case 0x06: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Marker %s", str); + cP[len] = c; + cP += len; + break; + case 0x05: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Lyric %s", str); + cP[len] = c; + cP += len; + break; + case 0x04: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Instrument Name %s", str); + cP[len] = c; + cP += len; + break; + case 0x03: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Sequence/Track Name %s", str); + cP[len] = c; + cP += len; + break; + case 0x02: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Copyright Notice %s", str); + cP[len] = c; + cP += len; + break; + case 0x01: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + msgPtr += sprintf(msgPtr, "Text %s", str); + cP[len] = c; + cP += len; + break; + case 0x00: + tt[0] = *cP++; + tt[1] = *cP++; + sn = midifile_get_multibyte_2(tt); + msgPtr += sprintf(msgPtr, "Sequence Number %d", sn); + break; + default: + msgPtr += sprintf(msgPtr, "Unknown: 0x%02X", c); + cP += len; + break; + } + break; + default: /* 0xF4, 0xF5, 0xF9, 0xFD are not defined */ + msgPtr += sprintf(msgPtr, "Undefined: 0x%02X", status); + break; + } + } + else + { + if (status & 0x80) + { + running_status = status; + c = *cP++; + } + else + { + c = status; + status = running_status; + } + ch = (status & 0x0F) + 1; /* MIDI channel number */ + switch (status & 0xF0) + { + case 0x80: + d = *cP++; /* 2 data bytes */ + msgPtr += sprintf(msgPtr, + "MIDI 0x%02X %02X %02X : channel %d Note %d Off velocity %d", + status, c, d, ch, c, d); + break; + case 0x90: + d = *cP++; /* 2 data bytes */ + if (d == 0) + msgPtr += sprintf(msgPtr,"MIDI 0x%02X %02X %02X : channel %d Note %d Off", status, c, d, ch, c); + else + msgPtr += sprintf(msgPtr, + "MIDI 0x%02X %02X %02X : channel %d Note %d On velocity %d", status, c, d, ch, c, d); + break; + case 0xA0: + d = *cP++; /* 2 data bytes */ + msgPtr += sprintf(msgPtr, + "MIDI: 0x%02X %02X %02X : channel %d Note %d Aftertouch %d", status, c, d, ch, c, d); + break; + case 0xB0: + d = *cP++; /* 2 data bytes */ + msgPtr += sprintf(msgPtr, + "MIDI: 0x%02X %02X %02X : channel %d Controller %d: %d", status, c, d, ch, c, d); + break; + case 0xC0: /* 1 data byte */ + msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X: channel %d Program Change: %d", status, c, ch, c); + break; + case 0xD0: /* 1 data byte */ + msgPtr += sprintf(msgPtr,"MIDI: 0x%02X %02X: channel %d Channel Pressure: %d", status, c, ch, c); + break; + case 0xE0: /* 2 data bytes */ + d = *cP++; /* 2 data bytes */ + msgPtr += sprintf(msgPtr, + "MIDI: 0x%02X %02X %02X : channel %d Pitch Wheel %d", + status, c, d, ch, midifile_combine_bytes(c, d)); + break; + } + } + post("midifile: %s", msg); + } } void midifile_get_next_track_chunk_data(t_midifile *x, int index) /* parse the next track chunk data element and output via the appropriate outlet or post to main window */ /* Sets the delta_time of the element or NO_MORE_ELEMENTS if no more elements */ { - unsigned char *cP, *last_cP, *str; - size_t delta_time, time_sig, len, i; - unsigned char status, c, d, nn, dd, cc, bb, mi, mcp, n; - char sf; - unsigned short sn; - unsigned char tt[3]; - - cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; - last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; - -// if (x->verbosity) post("Parsing track[%d]: ", index); - delta_time = NO_MORE_ELEMENTS; - if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) - { - cP = midifile_read_var_len(cP, &delta_time); - status = *cP++; -// if (x->verbosity) post ("delta time %lu total_time %lu status %02X", delta_time, x->track_chunk[index].total_time, status); - if ((status & 0xF0) == 0xF0) - { - switch (status) - {// system message - case 0xF0: - case 0xF7: - cP = midifile_read_var_len(cP, &len); // packet length - if (x->verbosity) post("midifile: Sysex: %02X length %lu", status, len); - midifile_output_long_list(x->midi_list_outlet, cP, len, 0xF0); - /* - outlet_float(x->sysex_outlet, 0xF0); // sysex status - for (i = 0; i < len; ++i) outlet_float(x->sysex_outlet, cP[i]); // sysex packet - //if (cP[len-1] != 0xF7)outlet_float(x->sysex_outlet, 0xF0); // sysex terminator - */ - cP += len; - x->track_chunk[index].running_status = 0; - break; - case 0xF1: // quarter frame - x->midi_data[0].a_w.w_float = status; - outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); - x->track_chunk[index].running_status = 0; - break; - case 0xF3: // song select - c = *cP++; - x->midi_data[0].a_w.w_float = status; - x->midi_data[1].a_w.w_float = c; - outlet_list(x->midi_list_outlet, &s_list, 2, x->midi_data); - x->track_chunk[index].running_status = 0; - break; - case 0xF2: // song position - c = *cP++; - x->midi_data[0].a_w.w_float = status; - x->midi_data[1].a_w.w_float = c; - c = *cP++; - x->midi_data[2].a_w.w_float = c; - outlet_list(x->midi_list_outlet, &s_list, 3, x->midi_data); - x->track_chunk[index].running_status = 0; - break; - case 0xF6: // tune request - x->midi_data[0].a_w.w_float = status; - outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); - x->track_chunk[index].running_status = 0; - break; - case 0xF8: // MIDI clock - case 0xF9: // MIDI tick - case 0xFA: // MIDI start - case 0xFB: // MIDI continue - case 0xFC: // MIDI stop - case 0xFE: // active sense - x->midi_data[0].a_w.w_float = status; - outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); - break; - case 0xFF: - c = *cP++; - cP = midifile_read_var_len(cP, &len);// meta length - if (x->verbosity) post("midifile: Track %d Meta: %02X length %lu", index, c, len); - switch (c) - { - case 0x58: - nn = *cP++; - dd = *cP++; - dd = 1<<(dd); - cc = *cP++; - bb = *cP++; - if (x->verbosity) post ("midifile: Time Signature: %d/%d %d clocks per tick, %d 32nd notes per quarternote", nn, dd, cc, bb); - break; - case 0x59: - sf = *(signed char *)cP++; - mi = *cP++; - if (x->verbosity) post ("midifile: Key Signature: %d %s, %s", sf, (sf<0)?"flats":"sharps", (mi)?"minor":"major"); - break; - case 0x51: - tt[0] = *cP++; - tt[1] = *cP++; - tt[2] = *cP++; - time_sig = midifile_get_multibyte_3(tt); - if (x->verbosity) post ("midifile: %lu microseconds per MIDI quarter-note", time_sig); - break; - case 0x2F: - if (x->verbosity) post ("midifile: End of Track %d", index); - delta_time = NO_MORE_ELEMENTS; - cP += len; - break; - case 0x21: - tt[0] = *cP++; - if (x->verbosity) post ("midifile: MIDI port or cable number (unofficial): %d", tt[0]); - break; - case 0x20: - mcp = *cP++; - if (x->verbosity) post ("midifile: MIDI Channel Prefix: %d", mcp); - break; - case 0x06: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Marker: %s", str); - cP[len] = c; - cP += len; - break; - case 0x05: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Lyric: %s", str); - cP[len] = c; - cP += len; - break; - case 0x04: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Instrument Name: %s", str); - cP[len] = c; - cP += len; - break; - case 0x03: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Sequence/Track Name: %s", str); - cP[len] = c; - cP += len; - break; - case 0x02: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Copyright Notice: %s", str); - cP[len] = c; - cP += len; - break; - case 0x01: - str = cP; - c = cP[len]; - cP[len] = '\0'; //null terminate temporarily - if (x->verbosity) post ("midifile: Text Event: %s", str); - cP[len] = c; - cP += len; - break; - case 0x00: - tt[0] = *cP++; - tt[1] = *cP++; - sn = midifile_get_multibyte_2(tt); - if (x->verbosity) post ("midifile: Sequence Number %d", sn); - break; - default: - if (x->verbosity) post ("midifile: Unknown: %02X", c); - cP += len; - break; - } - break; - default: // 0xF4, 0xF5, 0xF9, 0xFD are not defined - break; - } - } - else - { - if (status & 0x80) - { - x->track_chunk[index].running_status = status;// status is true status - c = *cP++; - } - else - { - c = status; // status is actually 1st data byte - status = x->track_chunk[index].running_status;// current status - } - switch (status & 0xF0) - { - case 0x80: - case 0x90: - case 0xA0: - case 0xB0: - case 0xE0: - n = 3; - d = *cP++; // 2 data bytes - break; - case 0xC0: // 1 data byte - case 0xD0: - n = 2; - break; - } - x->midi_data[0].a_w.w_float = status; - x->midi_data[1].a_w.w_float = c; - x->midi_data[2].a_w.w_float = (n == 3)?d:0; - if (x->midi_data[0].a_w.w_float != 0) outlet_list(x->midi_list_outlet, &s_list, n, x->midi_data); - //if (x->verbosity) post("MIDI: %02X %d %d", x->track_chunk[index].running_status, c, d); - if (x->track_chunk[index].running_status == 0) - error ("midifile: No running status on track %d at %lu", index, x->track_chunk[index].total_time + delta_time); - } - } - x->track_chunk[index].track_index = (char *)cP - (char *)x->track_chunk[index].track_data; - x->track_chunk[index].delta_time = delta_time; - if (delta_time == NO_MORE_ELEMENTS) x->track_chunk[index].total_time = delta_time; - else x->track_chunk[index].total_time += delta_time; + unsigned char *cP, *last_cP, *str; + size_t delta_time, time_sig, len, i; + unsigned char status, c, d, nn, dd, cc, bb, mi, mcp, n; + char sf; + unsigned short sn; + unsigned char tt[3]; + + cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; + last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; + + delta_time = NO_MORE_ELEMENTS; + if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) + { + cP = midifile_read_var_len(cP, &delta_time); + status = *cP++; + if ((status & 0xF0) == 0xF0) + { + switch (status) + { /* system message */ + case 0xF0: + case 0xF7: + cP = midifile_read_var_len(cP, &len); /* packet length */ + if (x->verbosity) post("midifile: Sysex: %02X length %lu", status, len); + midifile_output_long_list(x->midi_list_outlet, cP, len, 0xF0); + cP += len; + x->track_chunk[index].running_status = 0; + break; + case 0xF1: /* quarter frame */ + x->midi_data[0].a_w.w_float = status; + outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); + x->track_chunk[index].running_status = 0; + break; + case 0xF3: /* song select */ + c = *cP++; + x->midi_data[0].a_w.w_float = status; + x->midi_data[1].a_w.w_float = c; + outlet_list(x->midi_list_outlet, &s_list, 2, x->midi_data); + x->track_chunk[index].running_status = 0; + break; + case 0xF2: /* song position */ + c = *cP++; + x->midi_data[0].a_w.w_float = status; + x->midi_data[1].a_w.w_float = c; + c = *cP++; + x->midi_data[2].a_w.w_float = c; + outlet_list(x->midi_list_outlet, &s_list, 3, x->midi_data); + x->track_chunk[index].running_status = 0; + break; + case 0xF6: /* tune request */ + x->midi_data[0].a_w.w_float = status; + outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); + x->track_chunk[index].running_status = 0; + break; + case 0xF8: /* MIDI clock */ + case 0xF9: /* MIDI tick */ + case 0xFA: /* MIDI start */ + case 0xFB: /* MIDI continue */ + case 0xFC: /* MIDI stop */ + case 0xFE: /* active sense */ + x->midi_data[0].a_w.w_float = status; + outlet_list(x->midi_list_outlet, &s_list, 1, x->midi_data); + break; + case 0xFF: + c = *cP++; + cP = midifile_read_var_len(cP, &len);/* meta length */ + if (x->verbosity) post("midifile: Track %d Meta: %02X length %lu", index, c, len); + switch (c) + { + case 0x58: + nn = *cP++; + dd = *cP++; + dd = 1<<(dd); + cc = *cP++; + bb = *cP++; + if (x->verbosity) + post ("midifile: Time Signature: %d/%d %d clocks per tick, %d 32nd notes per quarternote", + nn, dd, cc, bb); + break; + case 0x59: + sf = *(signed char *)cP++; + mi = *cP++; + if (x->verbosity) + post ("midifile: Key Signature: %d %s, %s", + sf, (sf<0)?"flats":"sharps", (mi)?"minor":"major"); + break; + case 0x51: + tt[0] = *cP++; + tt[1] = *cP++; + tt[2] = *cP++; + time_sig = midifile_get_multibyte_3(tt); + if (x->verbosity) post ("midifile: %lu microseconds per MIDI quarter-note", time_sig); + break; + case 0x2F: + if (x->verbosity) post ("midifile: End of Track %d", index); + delta_time = NO_MORE_ELEMENTS; + cP += len; + break; + case 0x21: + tt[0] = *cP++; + if (x->verbosity) post ("midifile: MIDI port or cable number (unofficial): %d", tt[0]); + break; + case 0x20: + mcp = *cP++; + if (x->verbosity) post ("midifile: MIDI Channel Prefix: %d", mcp); + break; + case 0x06: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Marker: %s", str); + cP[len] = c; + cP += len; + break; + case 0x05: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Lyric: %s", str); + cP[len] = c; + cP += len; + break; + case 0x04: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Instrument Name: %s", str); + cP[len] = c; + cP += len; + break; + case 0x03: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Sequence/Track Name: %s", str); + cP[len] = c; + cP += len; + break; + case 0x02: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Copyright Notice: %s", str); + cP[len] = c; + cP += len; + break; + case 0x01: + str = cP; + c = cP[len]; + cP[len] = '\0'; /* null terminate temporarily */ + if (x->verbosity) post ("midifile: Text Event: %s", str); + cP[len] = c; + cP += len; + break; + case 0x00: + tt[0] = *cP++; + tt[1] = *cP++; + sn = midifile_get_multibyte_2(tt); + if (x->verbosity) post ("midifile: Sequence Number %d", sn); + break; + default: + if (x->verbosity) post ("midifile: Unknown: %02X", c); + cP += len; + break; + } + break; + default: /* 0xF4, 0xF5, 0xF9, 0xFD are not defined */ + break; + } + } + else + { + if (status & 0x80) + { + x->track_chunk[index].running_status = status;/* status is true status */ + c = *cP++; + } + else + { + c = status; /* status is actually 1st data byte */ + status = x->track_chunk[index].running_status; /* current status */ + } + switch (status & 0xF0) + { + case 0x80: + case 0x90: + case 0xA0: + case 0xB0: + case 0xE0: + n = 3; + d = *cP++; /* 2 data bytes */ + break; + case 0xC0: /* 1 data byte */ + case 0xD0: + n = 2; + break; + } + x->midi_data[0].a_w.w_float = status; + x->midi_data[1].a_w.w_float = c; + x->midi_data[2].a_w.w_float = (n == 3)?d:0; + if (x->midi_data[0].a_w.w_float != 0) outlet_list(x->midi_list_outlet, &s_list, n, x->midi_data); + if (x->track_chunk[index].running_status == 0) + error ("midifile: No running status on track %d at %lu", + index, x->track_chunk[index].total_time + delta_time); + } + } + x->track_chunk[index].track_index = (char *)cP - (char *)x->track_chunk[index].track_data; + x->track_chunk[index].delta_time = delta_time; + if (delta_time == NO_MORE_ELEMENTS) x->track_chunk[index].total_time = delta_time; + else x->track_chunk[index].total_time += delta_time; } void midifile_skip_next_track_chunk_data(t_midifile *x, int index) /* parse the next track chunk data element and skip it without any output */ /* Sets the delta_time of the element or NO_MORE_ELEMENTS if no more elements */ { - unsigned char *cP, *last_cP; - size_t delta_time, len; - unsigned char status, c, n; - - cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; - last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; - -// if (x->verbosity) post("Parsing track[%d]: ", index); - delta_time = NO_MORE_ELEMENTS; - - if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) - { - cP = midifile_read_var_len(cP, &delta_time); - status = *cP++; -// if (x->verbosity) post ("delta time %lu total_time %lu status %02X", delta_time, x->track_chunk[index].total_time, status); - if ((status & 0xF0) == 0xF0) - { - switch (status) - {// system message - case 0xF0: - case 0xF7: - cP = midifile_read_var_len(cP, &len); // packet length - cP += len; - break; - case 0xF1: // quarter frame - break; - case 0xF3: // song select - cP += 1; - break; - case 0xF2: // song position - cP += 2; - break; - case 0xF6: // tune request - case 0xF8: // MIDI clock - case 0xF9: // MIDI tick - case 0xFA: // MIDI start - case 0xFB: // MIDI continue - case 0xFC: // MIDI stop - case 0xFE: // active sense - break; - case 0xFF: - c = *cP++; - cP = midifile_read_var_len(cP, &len);// meta length - // if (x->verbosity) post("Meta: %02X length %lu", c, len); - switch (c) - { - case 0x2F: - if (x->verbosity) post ("midifile: End of Track %d", index); - delta_time = NO_MORE_ELEMENTS; - // fall through to default.... - default: - cP += len; - break; - } - break; - default: // 0xF4, 0xF5, 0xF9, 0xFD are not defined - break; - } - } - else - { - if (status & 0x80) - { - x->track_chunk[index].running_status = status; - n = 1; - } - else - { - n = 0; // no status in this message - status = x->track_chunk[index].running_status; - } - switch (status & 0xF0) - { - case 0x80: - case 0x90: - case 0xA0: - case 0xB0: - case 0xE0: - n += 1; // data bytes - break; - case 0xC0: - case 0xD0:// only one data byte - break; - } - cP += n; - } - } - x->track_chunk[index].track_index = (char *)cP - (char *)x->track_chunk[index].track_data; - x->track_chunk[index].delta_time = delta_time; - if (delta_time == NO_MORE_ELEMENTS) x->track_chunk[index].total_time = delta_time; - else x->track_chunk[index].total_time += delta_time; + unsigned char *cP, *last_cP; + size_t delta_time, len; + unsigned char status, c, n; + + cP = x->track_chunk[index].track_data + x->track_chunk[index].track_index; + last_cP = x->track_chunk[index].track_data + x->track_chunk[index].chunk_length; + + delta_time = NO_MORE_ELEMENTS; + + if ((cP != NULL) && (cP < last_cP) && (x->track_chunk[index].delta_time != NO_MORE_ELEMENTS)) + { + cP = midifile_read_var_len(cP, &delta_time); + status = *cP++; + if ((status & 0xF0) == 0xF0) + { + switch (status) + { /* system message */ + case 0xF0: + case 0xF7: + cP = midifile_read_var_len(cP, &len); /* packet length */ + cP += len; + break; + case 0xF1: /* quarter frame */ + break; + case 0xF3: /* song select */ + cP += 1; + break; + case 0xF2: /* song position */ + cP += 2; + break; + case 0xF6: /* tune request */ + case 0xF8: /* MIDI clock */ + case 0xF9: /* MIDI tick */ + case 0xFA: /* MIDI start */ + case 0xFB: /* MIDI continue */ + case 0xFC: /* MIDI stop */ + case 0xFE: /* active sense */ + break; + case 0xFF: + c = *cP++; + cP = midifile_read_var_len(cP, &len);/* meta length */ + switch (c) + { + case 0x2F: + if (x->verbosity) post ("midifile: End of Track %d", index); + delta_time = NO_MORE_ELEMENTS; + /* fall through to default....*/ + default: + cP += len; + break; + } + break; + default: /* 0xF4, 0xF5, 0xF9, 0xFD are not defined */ + break; + } + } + else + { + if (status & 0x80) + { + x->track_chunk[index].running_status = status; + n = 1; + } + else + { + n = 0; /* no status in this message */ + status = x->track_chunk[index].running_status; + } + switch (status & 0xF0) + { + case 0x80: + case 0x90: + case 0xA0: + case 0xB0: + case 0xE0: + n += 1; /* data bytes */ + break; + case 0xC0: + case 0xD0: /* only one data byte */ + break; + } + cP += n; + } + } + x->track_chunk[index].track_index = (char *)cP - (char *)x->track_chunk[index].track_data; + x->track_chunk[index].delta_time = delta_time; + if (delta_time == NO_MORE_ELEMENTS) x->track_chunk[index].total_time = delta_time; + else x->track_chunk[index].total_time += delta_time; } /* fin midifile.c */ |