123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848 |
- #include <ctype.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <stdlib.h>
- #ifdef MSDOS
- #include <allegro.h>
- #endif
- #include "mmus2mid.h"
- #include "lprintf.h"
- #ifndef STANDALONE
- #include "m_swap.h"
- #include "z_zone.h"
- #endif
- #define last(e) ((UBYTE)((e) & 0x80))
- #define event_type(e) ((UBYTE)(((e) & 0x7F) >> 4))
- #define channel(e) ((UBYTE)((e) & 0x0F))
- typedef enum
- {
- RELEASE_NOTE,
- PLAY_NOTE,
- BEND_NOTE,
- SYS_EVENT,
- CNTL_CHANGE,
- UNKNOWN_EVENT1,
- SCORE_END,
- UNKNOWN_EVENT2,
- } mus_event_t;
- typedef struct
- {
- char ID[4];
- UWORD ScoreLength;
- UWORD ScoreStart;
- UWORD channels;
- UWORD SecChannels;
- UWORD InstrCnt;
- } PACKEDATTR MUSheader;
- typedef struct Track
- {
- char velocity;
- long deltaT;
- UBYTE lastEvt;
- long alloced;
- } TrackInfo;
- static TrackInfo track[MIDI_TRACKS];
- #define TRACKBUFFERSIZE 1024
- static UBYTE MUS2MIDcontrol[15] =
- {
- 0,
- 0x00,
- 0x01,
- 0x07,
- 0x0A,
- 0x0B,
- 0x5B,
- 0x5D,
- 0x40,
- 0x43,
- 0x78,
- 0x7B,
- 0x7E,
- 0x7F,
- 0x79
- };
- static UBYTE midikey[] =
- {0x00,0xff,0x59,0x02,0x00,0x00};
- static UBYTE miditempo[] =
- {0x00,0xff,0x51,0x03,0x09,0xa3,0x1a};
- static UBYTE midihdr[] =
- {'M','T','h','d',0,0,0,6,0,1,0,0,0,0};
- static UBYTE trackhdr[] =
- {'M','T','r','k'};
- static int TWriteByte(MIDI *mididata, int MIDItrack, UBYTE byte);
- static int TWriteVarLen(MIDI *mididata, int MIDItrack, register ULONG value);
- static ULONG ReadTime(const UBYTE **musptrp);
- static int FirstChannelAvailable(int MUS2MIDchannel[]);
- static UBYTE MidiEvent(MIDI *mididata,UBYTE midicode,UBYTE MIDIchannel,
- UBYTE MIDItrack,int nocomp);
- static int TWriteByte(MIDI *mididata, int MIDItrack, UBYTE byte)
- {
- ULONG pos ;
- pos = mididata->track[MIDItrack].len;
- if (pos >= (ULONG)track[MIDItrack].alloced)
- {
- track[MIDItrack].alloced =
- track[MIDItrack].alloced?
- 2*track[MIDItrack].alloced :
- TRACKBUFFERSIZE;
- if (!(mididata->track[MIDItrack].data =
- realloc(mididata->track[MIDItrack].data,
- track[MIDItrack].alloced)))
- return MEMALLOC;
- }
- mididata->track[MIDItrack].data[pos] = byte;
- mididata->track[MIDItrack].len++;
- return 0;
- }
- static int TWriteVarLen(MIDI *mididata, int tracknum, register ULONG value)
- {
- register ULONG buffer;
- buffer = value & 0x7f;
- while ((value >>= 7))
- {
- buffer <<= 8;
- buffer |= 0x80;
- buffer += (value & 0x7f);
- }
- while (1)
- {
- if (TWriteByte(mididata, tracknum, (UBYTE)(buffer&0xff)))
- return MEMALLOC;
- if (buffer & 0x80)
- buffer >>= 8;
- else
- break;
- }
- return 0;
- }
- static ULONG ReadTime(const UBYTE **musptrp)
- {
- register ULONG timeval = 0;
- int byte;
- do
- {
- byte = *(*musptrp)++;
- timeval = (timeval << 7) + (byte & 0x7F);
- }
- while(byte & 0x80);
- return timeval;
- }
- static int FirstChannelAvailable(int MUS2MIDchannel[])
- {
- int i ;
- int max = -1 ;
-
- for (i = 0; i < 15; i++)
- if (MUS2MIDchannel[i] > max)
- max = MUS2MIDchannel[i];
- return (max == 8 ? 10 : max+1);
- }
- static UBYTE MidiEvent(MIDI *mididata,UBYTE midicode,UBYTE MIDIchannel,
- UBYTE MIDItrack,int nocomp)
- {
- UBYTE newevent;
- newevent = midicode | MIDIchannel;
- if ((newevent != track[MIDItrack].lastEvt) || nocomp)
- {
- if (TWriteByte(mididata,MIDItrack, newevent))
- return 0;
- track[MIDItrack].lastEvt = newevent;
- }
- return newevent;
- }
- int mmus2mid(const UBYTE *mus, MIDI *mididata, UWORD division, int nocomp)
- {
- UWORD TrackCnt = 0;
- UBYTE evt, MUSchannel, MIDIchannel, MIDItrack=0, NewEvent;
- int i, event, data;
- const UBYTE *musptr;
- size_t muslen;
- static MUSheader MUSh;
- UBYTE MIDIchan2track[MIDI_TRACKS];
- int MUS2MIDchannel[MIDI_TRACKS];
-
- memcpy(&MUSh,mus,sizeof(MUSheader));
- MUSh.ScoreLength = doom_wtohs(MUSh.ScoreLength);
- MUSh.ScoreStart = doom_wtohs(MUSh.ScoreStart);
- MUSh.channels = doom_wtohs(MUSh.channels);
- MUSh.SecChannels = doom_wtohs(MUSh.SecChannels);
- MUSh.InstrCnt = doom_wtohs(MUSh.InstrCnt);
-
- if (!(muslen = MUSh.ScoreLength + MUSh.ScoreStart))
- return MUSDATAMT;
- if (MUSh.channels > 15)
- return TOOMCHAN ;
- musptr = mus+MUSh.ScoreStart;
- for (i = 0; i < MIDI_TRACKS; i++)
- {
- MUS2MIDchannel[i] = -1;
- track[i].velocity = 64;
- track[i].deltaT = 0;
- track[i].lastEvt = 0;
-
- mididata->track[i].data=NULL;
- track[i].alloced = 0;
- mididata->track[i].len = 0;
- }
- if (!division)
- division = 70;
-
-
-
- mididata->divisions = division;
-
- if (!(mididata->track[0].data =
- realloc(mididata->track[0].data,sizeof(midikey)+sizeof(miditempo)+4)))
- return MEMALLOC;
-
- memcpy(mididata->track[0].data,midikey,sizeof(midikey));
-
- memcpy(mididata->track[0].data+sizeof(midikey),miditempo,sizeof(miditempo));
- mididata->track[0].len = sizeof(midikey)+sizeof(miditempo);
- TrackCnt++;
-
- do
- {
-
- event = *musptr++;
- if ((evt = event_type(event)) == SCORE_END)
- break;
- MUSchannel = channel(event);
-
- if (MUS2MIDchannel[MUSchannel] == -1)
- {
-
- MIDIchannel = MUS2MIDchannel[MUSchannel] =
- (MUSchannel == 15 ? 9 : FirstChannelAvailable(MUS2MIDchannel));
- MIDItrack = MIDIchan2track[MIDIchannel] = (UBYTE)TrackCnt++;
- }
- else
- {
- MIDIchannel = MUS2MIDchannel[MUSchannel];
- MIDItrack = MIDIchan2track[MIDIchannel];
- }
- if (TWriteVarLen(mididata, MIDItrack, track[MIDItrack].deltaT))
- return MEMALLOC;
- track[MIDItrack].deltaT = 0;
- switch(evt)
- {
- case RELEASE_NOTE:
-
- if (!(NewEvent=MidiEvent(mididata,0x90,MIDIchannel,MIDItrack,1)))
- return MEMALLOC;
- data = *musptr++;
- if (TWriteByte(mididata, MIDItrack, (UBYTE)(data & 0x7F)))
- return MEMALLOC;
- if (TWriteByte(mididata, MIDItrack, 0))
- return MEMALLOC;
- break;
- case PLAY_NOTE:
- if (!(NewEvent=MidiEvent(mididata,0x90,MIDIchannel,MIDItrack,nocomp)))
- return MEMALLOC;
- data = *musptr++;
- if (TWriteByte(mididata, MIDItrack, (UBYTE)(data & 0x7F)))
- return MEMALLOC;
- if( data & 0x80 )
- track[MIDItrack].velocity = (*musptr++) & 0x7f;
- if (TWriteByte(mididata, MIDItrack, track[MIDItrack].velocity))
- return MEMALLOC;
- break;
- case BEND_NOTE:
- if (!(NewEvent=MidiEvent(mididata,0xE0,MIDIchannel,MIDItrack,nocomp)))
- return MEMALLOC;
- data = *musptr++;
- if (TWriteByte(mididata, MIDItrack, (UBYTE)((data & 1) << 6)))
- return MEMALLOC;
- if (TWriteByte(mididata, MIDItrack, (UBYTE)(data >> 1)))
- return MEMALLOC;
- break;
- case SYS_EVENT:
- if (!(NewEvent=MidiEvent(mididata,0xB0,MIDIchannel,MIDItrack,nocomp)))
- return MEMALLOC;
- data = *musptr++;
- if (data<10 || data>14)
- return BADSYSEVT;
- if (TWriteByte(mididata, MIDItrack, MUS2MIDcontrol[data]))
- return MEMALLOC;
- if (data == 12)
- {
- if (TWriteByte(mididata, MIDItrack, (UBYTE)(MUSh.channels+1)))
- return MEMALLOC;
- }
- else
- if (TWriteByte(mididata, MIDItrack, 0))
- return MEMALLOC;
- break;
- case CNTL_CHANGE:
- data = *musptr++;
- if (data>9)
- return BADCTLCHG;
- if (data)
- {
- if (!(NewEvent=MidiEvent(mididata,0xB0,MIDIchannel,MIDItrack,nocomp)))
- return MEMALLOC;
- if (TWriteByte(mididata, MIDItrack, MUS2MIDcontrol[data]))
- return MEMALLOC;
- }
- else
- {
- if (!(NewEvent=MidiEvent(mididata,0xC0,MIDIchannel,MIDItrack,nocomp)))
- return MEMALLOC;
- }
- data = *musptr++;
- if (TWriteByte(mididata, MIDItrack, (UBYTE)(data & 0x7F)))
- return MEMALLOC;
- break;
- case UNKNOWN_EVENT1:
- case UNKNOWN_EVENT2:
- return BADMUSCTL;
- case SCORE_END:
- break;
- default:
- return BADMUSCTL;
- }
- if (last(event))
- {
- ULONG DeltaTime = ReadTime(&musptr);
- for (i = 0;i < MIDI_TRACKS; i++)
- track[i].deltaT += DeltaTime;
- }
- }
- while ((evt != SCORE_END) && ((size_t)(musptr-mus) < muslen));
- if (evt!=SCORE_END)
- return MUSDATACOR;
-
- for (i = 0; i < MIDI_TRACKS; i++)
- if (mididata->track[i].len)
- {
- if (TWriteByte(mididata, i, 0x00) ||
- TWriteByte(mididata, i, 0xFF) ||
- TWriteByte(mididata, i, 0x2F) ||
- TWriteByte(mididata, i, 0x00))
- return MEMALLOC;
-
-
- if (!(mididata->track[i].data =
- realloc(mididata->track[i].data,mididata->track[i].len)))
- return MEMALLOC;
- }
- else
- {
- free(mididata->track[i].data);
- mididata->track[i].data = NULL;
- }
- return 0;
- }
- void free_mididata(MIDI *mid)
- {
- int i;
- for (i = 0; i < MIDI_TRACKS; i++)
- if (mid->track[i].data)
- free(mid->track[i].data);
- }
- static size_t ReadLength(UBYTE **mid)
- {
- UBYTE *midptr = *mid;
- size_t length = (*midptr++)<<24;
- length += (*midptr++)<<16;
- length += (*midptr++)<<8;
- length += *midptr++;
- *mid = midptr;
- return length;
- }
- int MidiToMIDI(UBYTE *mid,MIDI *mididata)
- {
- int i;
- int ntracks;
-
- if (memcmp(mid,midihdr,4))
- return BADMIDHDR;
- mididata->divisions = (mid[12]<<8)+mid[13];
- ntracks = (mid[10]<<8)+mid[11];
- if (ntracks>=MIDI_TRACKS)
- return BADMIDHDR;
- mid += 4;
- {
- size_t t = ReadLength(&mid);
- mid += t;
- }
-
- for (i=0;i<ntracks;i++)
- {
- while (memcmp(mid,trackhdr,4))
- {
- mid += 4;
- {
- size_t t = ReadLength(&mid);
- mid += t;
- }
- }
- mid += 4;
- mididata->track[i].len = ReadLength(&mid);
-
- mididata->track[i].data = realloc(mididata->track[i].data,mididata->track[i].len);
- memcpy(mididata->track[i].data,mid,mididata->track[i].len);
- mid += mididata->track[i].len;
- }
- for (;i<MIDI_TRACKS;i++)
- if (mididata->track[i].len)
- {
- free(mididata->track[i].data);
- mididata->track[i].data = NULL;
- mididata->track[i].len = 0;
- }
- return 0;
- }
- static void TWriteLength(UBYTE **midiptr,ULONG length);
- static void TWriteLength(UBYTE **midiptr,ULONG length)
- {
- *(*midiptr)++ = (unsigned char)((length>>24)&0xff);
- *(*midiptr)++ = (unsigned char)((length>>16)&0xff);
- *(*midiptr)++ = (unsigned char)((length>>8)&0xff);
- *(*midiptr)++ = (unsigned char)((length)&0xff);
- }
- int MIDIToMidi(MIDI *mididata,UBYTE **mid,int *midlen)
- {
- size_t total;
- int i,ntrks;
- UBYTE *midiptr;
-
- total = sizeof(midihdr);
- for (i=0,ntrks=0;i<MIDI_TRACKS;i++)
- if (mididata->track[i].len)
- {
- total += 8 + mididata->track[i].len;
- ntrks++;
- }
- if ((*mid = malloc(total))==NULL)
- return MEMALLOC;
-
- midihdr[10] = 0;
- midihdr[11] = (UBYTE)ntrks;
- midihdr[12] = (mididata->divisions>>8) & 0x7f;
- midihdr[13] = (mididata->divisions) & 0xff;
-
- midiptr = *mid;
- memcpy(midiptr,midihdr,sizeof(midihdr));
- midiptr += sizeof(midihdr);
-
- for (i=0;i<MIDI_TRACKS;i++)
- {
- if (mididata->track[i].len)
- {
- memcpy(midiptr,trackhdr,sizeof(trackhdr));
- midiptr += sizeof(trackhdr);
- TWriteLength(&midiptr,mididata->track[i].len);
-
- memcpy(midiptr,mididata->track[i].data,mididata->track[i].len);
- midiptr += mididata->track[i].len;
- }
- }
-
- *midlen = midiptr - *mid;
- return 0;
- }
- #ifdef STANDALONE
-
- int main(int argc,char **argv)
- {
- FILE *musst,*midst;
- char musfile[FILENAME_MAX],midfile[FILENAME_MAX];
- MUSheader MUSh;
- UBYTE *mus,*mid;
- static MIDI mididata;
- int err,midlen;
- char *p,*q;
- int i;
- if (argc<2)
- {
-
- lprintf(LO_INFO,"Usage: MMUS2MID musfile[.MUS]\n");
- lprintf(LO_INFO,"writes musfile.MID as output\n");
- lprintf(LO_INFO,"musfile may contain wildcards\n");
- exit(1);
- }
- for (i=1;i<argc;i++)
- {
- strcpy(musfile,argv[i]);
- p = strrchr(musfile,'.');
- q = strrchr(musfile,'\\');
- if (p && (!q || q<p)) *p='\0';
- strcpy(midfile,musfile);
- strcat(musfile,".MUS");
- strcat(midfile,".MID");
- musst = fopen(musfile,"rb");
- if (musst)
- {
- fread(&MUSh,sizeof(MUSheader),1,musst);
- mus = malloc(MUSh.ScoreLength+MUSh.ScoreStart);
- if (mus)
- {
- fseek(musst,0,SEEK_SET);
- if (!fread(mus,MUSh.ScoreLength+MUSh.ScoreStart,1,musst))
- {
-
- lprintf(LO_FATAL,"Error reading MUS file\n");
- free(mus);
- exit(1);
- }
- fclose(musst);
- }
- else
- {
-
- lprintf(LO_FATAL,"Out of memory\n");
- free(mus);
- exit(1);
- }
- err = mmus2mid(mus,&mididata,89,1);
- if (err)
- {
-
- lprintf(LO_FATAL,"Error converting MUS file to MIDI: %d\n",err);
- exit(1);
- }
- free(mus);
- MIDIToMidi(&mididata,&mid,&midlen);
- midst = fopen(midfile,"wb");
- if (midst)
- {
- if (!fwrite(mid,midlen,1,midst))
- {
-
- lprintf(LO_FATAL,"Error writing MIDI file\n");
- FreeTracks(&mididata);
- free(mid);
- exit(1);
- }
- fclose(midst);
- }
- else
- {
-
- lprintf(LO_FATAL,"Can't open MIDI file for output: %s\n", midfile);
- FreeTracks(&mididata);
- free(mid);
- exit(1);
- }
- }
- else
- {
-
- lprintf(LO_FATAL,"Can't open MUS file for input: %s\n", midfile);
- exit(1);
- }
-
- lprintf(LO_CONFIRM,"MUS file %s converted to MIDI file %s\n",musfile,midfile);
- FreeTracks(&mididata);
- free(mid);
- }
- exit(0);
- }
- #endif
|