libcm is a C development framework with an emphasis on audio signal processing applications.
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

cmMidiFile.c 45KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764
  1. #include "cmPrefix.h"
  2. #include "cmGlobal.h"
  3. #include "cmRpt.h"
  4. #include "cmErr.h"
  5. #include "cmCtx.h"
  6. #include "cmFile.h"
  7. #include "cmMem.h"
  8. #include "cmMallocDebug.h"
  9. #include "cmLinkedHeap.h"
  10. #include "cmTime.h"
  11. #include "cmMidi.h"
  12. #include "cmMidiFile.h"
  13. #include "cmFloatTypes.h"
  14. #include "cmVectOps.h"
  15. #ifdef cmBIG_ENDIAN
  16. #define mfSwap16(v) (v)
  17. #define mfSwap32(v) (v)
  18. #else
  19. #define mfSwap16(v) cmSwap16(v)
  20. #define mfSwap32(v) cmSwap32(v)
  21. #endif
  22. typedef struct
  23. {
  24. unsigned cnt; // count of track records
  25. cmMidiTrackMsg_t* base; // pointer to first track recd
  26. cmMidiTrackMsg_t* last; // pointer to last track recd
  27. } _cmMidiTrack_t;
  28. typedef struct
  29. {
  30. cmErr_t err; // this objects error object
  31. cmLHeapH_t lhH; // linked heap used for all dynamically alloc'd data space
  32. cmFileH_t fh; // cmFile handle (only used in fmMidiFileOpen() and cmMidiFileWrite())
  33. unsigned short fmtId; // midi file type id: 0,1,2
  34. unsigned short ticksPerQN; // ticks per quarter note or 0 if smpteFmtId is valid
  35. cmMidiByte_t smpteFmtId; // smpte format or 0 if ticksPerQN is valid
  36. cmMidiByte_t smpteTicksPerFrame; // smpte ticks per frame or 0 if ticksPerQN is valid
  37. unsigned short trkN; // track count
  38. _cmMidiTrack_t* trkV; // track vector
  39. char* fn; // file name or NULL if this object did not originate from a file
  40. unsigned msgN; // count of msg's in msgV[]
  41. cmMidiTrackMsg_t** msgV; // sorted msg list
  42. unsigned nextUid; // next available msg uid
  43. } _cmMidiFile_t;
  44. cmMidiFileH_t cmMidiFileNullHandle = cmSTATIC_NULL_HANDLE;
  45. _cmMidiFile_t* _cmMidiFileHandleToPtr( cmMidiFileH_t h )
  46. {
  47. _cmMidiFile_t* p = (_cmMidiFile_t*)h.h;
  48. assert( p != NULL );
  49. return p;
  50. }
  51. void* _cmMidiFileMalloc( _cmMidiFile_t* mfp, unsigned byteN )
  52. { return cmLHeapAllocZ(mfp->lhH,byteN); }
  53. cmMfRC_t _cmMidiFileRead8( _cmMidiFile_t* mfp, cmMidiByte_t* p )
  54. {
  55. if( cmFileReadUChar(mfp->fh,p,1) != kOkFileRC )
  56. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI byte read failed.");
  57. return kOkMfRC;
  58. }
  59. cmMfRC_t _cmMidiFileRead16( _cmMidiFile_t* mfp, unsigned short* p )
  60. {
  61. if( cmFileReadUShort(mfp->fh,p,1) != kOkFileRC )
  62. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI short read failed.");
  63. *p = mfSwap16(*p);
  64. return kOkMfRC;
  65. }
  66. cmMfRC_t _cmMidiFileRead24( _cmMidiFile_t* mfp, unsigned* p )
  67. {
  68. *p = 0;
  69. int i = 0;
  70. for(; i<3; ++i)
  71. {
  72. unsigned char c;
  73. if( cmFileReadUChar(mfp->fh,&c,1) != kOkFileRC )
  74. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI 24 bit integer read failed.");
  75. *p = (*p << 8) + c;
  76. }
  77. //*p =mfSwap32(*p);
  78. return kOkMfRC;
  79. }
  80. cmMfRC_t _cmMidiFileRead32( _cmMidiFile_t* mfp, unsigned* p )
  81. {
  82. if( cmFileReadUInt(mfp->fh,p,1) != kOkFileRC )
  83. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI integer read failed.");
  84. *p = mfSwap32(*p);
  85. return kOkMfRC;
  86. }
  87. cmMfRC_t _cmMidiFileReadText( _cmMidiFile_t* mfp, cmMidiTrackMsg_t* tmp, unsigned byteN )
  88. {
  89. if( byteN == 0 )
  90. return kOkMfRC;
  91. char* t = (char*)_cmMidiFileMalloc(mfp,byteN+1);
  92. t[byteN] = 0;
  93. if( cmFileReadChar(mfp->fh,t,byteN) != kOkFileRC )
  94. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI read text failed.");
  95. tmp->u.text = t;
  96. tmp->byteCnt = byteN;
  97. return kOkMfRC;
  98. }
  99. cmMfRC_t _cmMidiFileReadRecd( _cmMidiFile_t* mfp, cmMidiTrackMsg_t* tmp, unsigned byteN )
  100. {
  101. char* t = (char*)_cmMidiFileMalloc(mfp,byteN);
  102. if( cmFileReadChar(mfp->fh,t,byteN) != kOkFileRC )
  103. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI read record failed.");
  104. tmp->byteCnt = byteN;
  105. tmp->u.voidPtr = t;
  106. return kOkMfRC;
  107. }
  108. cmMfRC_t _cmMidiFileReadVarLen( _cmMidiFile_t* mfp, unsigned* p )
  109. {
  110. unsigned char c;
  111. if( cmFileReadUChar(mfp->fh,&c,1) != kOkFileRC )
  112. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI read variable length integer failed.");
  113. if( !(c & 0x80) )
  114. *p = c;
  115. else
  116. {
  117. *p = c & 0x7f;
  118. do
  119. {
  120. if( cmFileReadUChar(mfp->fh,&c,1) != kOkFileRC )
  121. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI read variable length integer failed.");
  122. *p = (*p << 7) + (c & 0x7f);
  123. }while( c & 0x80 );
  124. }
  125. return kOkMfRC;
  126. }
  127. cmMfRC_t _cmMidiFileAppendTrackMsg( _cmMidiFile_t* mfp, unsigned short trkIdx, unsigned dtick, cmMidiByte_t status, cmMidiTrackMsg_t** trkMsgPtrPtr )
  128. {
  129. cmMidiTrackMsg_t* tmp = (cmMidiTrackMsg_t*)_cmMidiFileMalloc(mfp, sizeof(cmMidiTrackMsg_t) );
  130. // link new record onto track record chain
  131. if( mfp->trkV[trkIdx].base == NULL )
  132. mfp->trkV[trkIdx].base = tmp;
  133. else
  134. mfp->trkV[trkIdx].last->link = tmp;
  135. mfp->trkV[trkIdx].last = tmp;
  136. mfp->trkV[trkIdx].cnt++;
  137. // set the generic track record fields
  138. tmp->dtick = dtick;
  139. tmp->status = status;
  140. tmp->metaId = kInvalidMetaMdId;
  141. tmp->trkIdx = trkIdx;
  142. tmp->byteCnt = 0;
  143. *trkMsgPtrPtr = tmp;
  144. return kOkMfRC;
  145. }
  146. cmMfRC_t _cmMidiFileReadSysEx( _cmMidiFile_t* mfp, cmMidiTrackMsg_t* tmp, unsigned byteN )
  147. {
  148. cmMfRC_t rc = kOkMfRC;
  149. cmMidiByte_t b = 0;
  150. if( byteN == cmInvalidCnt )
  151. {
  152. long offs;
  153. if( cmFileTell(mfp->fh,&offs) != kOkFileRC )
  154. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI File 'tell' failed.");
  155. byteN = 0;
  156. // get the length of the sys-ex msg
  157. while( !cmFileEof(mfp->fh) && (b != kSysComEoxMdId) )
  158. {
  159. if((rc = _cmMidiFileRead8(mfp,&b)) != kOkMfRC )
  160. return rc;
  161. ++byteN;
  162. }
  163. // verify that the EOX byte was found
  164. if( b != kSysComEoxMdId )
  165. return cmErrMsg(&mfp->err,kMissingEoxMfRC,"MIDI file missing 'end-of-sys-ex'.");
  166. // rewind to the beginning of the msg
  167. if( cmFileSeek(mfp->fh,kBeginFileFl,offs) != kOkFileRC )
  168. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file seek failed on sys-ex read.");
  169. }
  170. // allocate memory to hold the sys-ex msg
  171. cmMidiByte_t* mp = (cmMidiByte_t *)_cmMidiFileMalloc(mfp, byteN );
  172. // read the sys-ex msg from the file into msg memory
  173. if( cmFileReadUChar(mfp->fh,mp,byteN) != kOkFileRC )
  174. return cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI sys-ex read failed.");
  175. tmp->byteCnt = byteN;
  176. tmp->u.sysExPtr = mp;
  177. return rc;
  178. }
  179. cmMfRC_t _cmMidiFileReadChannelMsg( _cmMidiFile_t* mfp, cmMidiByte_t* rsPtr, cmMidiByte_t status, cmMidiTrackMsg_t* tmp )
  180. {
  181. cmMfRC_t rc = kOkMfRC;
  182. cmMidiChMsg_t* p = (cmMidiChMsg_t*)_cmMidiFileMalloc(mfp,sizeof(cmMidiChMsg_t));
  183. unsigned useRsFl = status <= 0x7f;
  184. cmMidiByte_t statusCh = useRsFl ? *rsPtr : status;
  185. if( useRsFl )
  186. p->d0 = status;
  187. else
  188. *rsPtr = status;
  189. tmp->byteCnt = sizeof(cmMidiChMsg_t);
  190. tmp->status = statusCh & 0xf0;
  191. p->ch = statusCh & 0x0f;
  192. p->durMicros = 0;
  193. unsigned byteN = cmMidiStatusToByteCount(tmp->status);
  194. if( byteN==kInvalidMidiByte || byteN > 2 )
  195. return cmErrMsg(&mfp->err,kInvalidStatusMfRC,"Invalid status:0x%x %i.",tmp->status,tmp->status);
  196. unsigned i;
  197. for(i=useRsFl; i<byteN; ++i)
  198. {
  199. cmMidiByte_t* b = i==0 ? &p->d0 : &p->d1;
  200. if((rc = _cmMidiFileRead8(mfp,b)) != kOkMfRC )
  201. return rc;
  202. }
  203. // convert note-on velocity=0 to note off
  204. if( tmp->status == kNoteOnMdId && p->d1==0 )
  205. tmp->status = kNoteOffMdId;
  206. tmp->u.chMsgPtr = p;
  207. return rc;
  208. }
  209. cmMfRC_t _cmMidiFileReadMetaMsg( _cmMidiFile_t* mfp, cmMidiTrackMsg_t* tmp )
  210. {
  211. cmMidiByte_t metaId;
  212. cmMfRC_t rc;
  213. unsigned byteN = 0;
  214. if((rc = _cmMidiFileRead8(mfp,&metaId)) != kOkMfRC )
  215. return rc;
  216. if((rc = _cmMidiFileReadVarLen(mfp,&byteN)) != kOkMfRC )
  217. return rc;
  218. //printf("mt: %i 0x%x n:%i\n",metaId,metaId,byteN);
  219. switch( metaId )
  220. {
  221. case kSeqNumbMdId: rc = _cmMidiFileRead16(mfp,&tmp->u.sVal); break;
  222. case kTextMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  223. case kCopyMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  224. case kTrkNameMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  225. case kInstrNameMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  226. case kLyricsMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  227. case kMarkerMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  228. case kCuePointMdId: rc = _cmMidiFileReadText(mfp,tmp,byteN); break;
  229. case kMidiChMdId: rc = _cmMidiFileRead8(mfp,&tmp->u.bVal); break;
  230. case kEndOfTrkMdId: break;
  231. case kTempoMdId: rc = _cmMidiFileRead24(mfp,&tmp->u.iVal); break;
  232. case kSmpteMdId: rc = _cmMidiFileReadRecd(mfp,tmp,sizeof(cmMidiSmpte_t)); break;
  233. case kTimeSigMdId: rc = _cmMidiFileReadRecd(mfp,tmp,sizeof(cmMidiTimeSig_t)); break;
  234. case kKeySigMdId: rc = _cmMidiFileReadRecd(mfp,tmp,sizeof(cmMidiKeySig_t)); break;
  235. case kSeqSpecMdId: rc = _cmMidiFileReadSysEx(mfp,tmp,byteN); break;
  236. default:
  237. cmFileSeek(mfp->fh,kCurFileFl,byteN);
  238. rc = cmErrMsg(&mfp->err,kUnknownMetaIdMfRC,"Unknown meta status:0x%x %i.",metaId,metaId);
  239. }
  240. tmp->metaId = metaId;
  241. return rc;
  242. }
  243. cmMfRC_t _cmMidiFileReadTrack( _cmMidiFile_t* mfp, unsigned short trkIdx )
  244. {
  245. cmMfRC_t rc = kOkMfRC;
  246. unsigned dticks = 0;
  247. cmMidiByte_t status;
  248. cmMidiByte_t runstatus = 0;
  249. bool contFl = true;
  250. while( contFl && (rc==kOkMfRC))
  251. {
  252. cmMidiTrackMsg_t* tmp = NULL;
  253. // read the tick count
  254. if((rc = _cmMidiFileReadVarLen(mfp,&dticks)) != kOkMfRC )
  255. return rc;
  256. // read the status byte
  257. if((rc = _cmMidiFileRead8(mfp,&status)) != kOkMfRC )
  258. return rc;
  259. //printf("st:%i 0x%x\n",status,status);
  260. // append a track msg
  261. if((rc = _cmMidiFileAppendTrackMsg( mfp, trkIdx, dticks, status, &tmp )) != kOkMfRC )
  262. return rc;
  263. // switch on status
  264. switch( status )
  265. {
  266. // handle sys-ex msg
  267. case kSysExMdId:
  268. rc = _cmMidiFileReadSysEx(mfp,tmp,cmInvalidCnt);
  269. break;
  270. // handle meta msg
  271. case kMetaStId:
  272. rc = _cmMidiFileReadMetaMsg(mfp,tmp);
  273. // ignore unknown meta messages
  274. if( rc == kUnknownMetaIdMfRC )
  275. rc = kOkMfRC;
  276. contFl = tmp->metaId != kEndOfTrkMdId;
  277. break;
  278. default:
  279. // handle channel msg
  280. rc = _cmMidiFileReadChannelMsg(mfp,&runstatus,status,tmp);
  281. }
  282. }
  283. return rc;
  284. }
  285. cmMfRC_t _cmMidiFileReadHdr( _cmMidiFile_t* mfp )
  286. {
  287. cmMfRC_t rc;
  288. unsigned fileId;
  289. unsigned chunkByteN;
  290. // read the file id
  291. if((rc = _cmMidiFileRead32(mfp,&fileId)) != kOkMfRC )
  292. return rc;
  293. // verify the file id
  294. if( fileId != 'MThd' )
  295. return cmErrMsg(&mfp->err,kNotAMidiFileMfRC,"");
  296. // read the file chunk byte count
  297. if((rc = _cmMidiFileRead32(mfp,&chunkByteN)) != kOkMfRC )
  298. return rc;
  299. // read the format id
  300. if((rc = _cmMidiFileRead16(mfp,&mfp->fmtId)) != kOkMfRC )
  301. return rc;
  302. // read the track count
  303. if((rc = _cmMidiFileRead16(mfp,&mfp->trkN)) != kOkMfRC )
  304. return rc;
  305. // read the ticks per quarter note
  306. if((rc = _cmMidiFileRead16(mfp,&mfp->ticksPerQN)) != kOkMfRC )
  307. return rc;
  308. // if the division field was given in smpte
  309. if( mfp->ticksPerQN & 0x8000 )
  310. {
  311. mfp->smpteFmtId = (mfp->ticksPerQN & 0x7f00) >> 8;
  312. mfp->smpteTicksPerFrame = (mfp->ticksPerQN & 0xFF);
  313. mfp->ticksPerQN = 0;
  314. }
  315. // allocate and zero the track array
  316. if( mfp->trkN )
  317. mfp->trkV = _cmMidiFileMalloc( mfp, sizeof(_cmMidiTrack_t)*mfp->trkN);
  318. return rc;
  319. }
  320. int _cmMidiFileSortFunc( const void *p0, const void* p1 )
  321. {
  322. if( (*(cmMidiTrackMsg_t**)p0)->atick == (*(cmMidiTrackMsg_t**)p1)->atick )
  323. return 0;
  324. return (*(cmMidiTrackMsg_t**)p0)->atick < (*(cmMidiTrackMsg_t**)p1)->atick ? -1 : 1;
  325. }
  326. // Set the absolute accumulated ticks (atick) value of each track message.
  327. // The absolute accumulated ticks gives a global time ordering for all
  328. // messages in the file.
  329. void _cmMidiFileSetAccumulateTicks( _cmMidiFile_t* p )
  330. {
  331. cmMidiTrackMsg_t* nextTrkMsg[ p->trkN ]; // next msg in each track
  332. unsigned long long atick = 0;
  333. unsigned i;
  334. bool fl = true;
  335. // iniitalize nextTrkTick[] and nextTrkMsg[].
  336. for(i=0; i<p->trkN; ++i)
  337. if((nextTrkMsg[i] = p->trkV[i].base) != NULL )
  338. nextTrkMsg[i]->atick = nextTrkMsg[i]->dtick;
  339. while(1)
  340. {
  341. unsigned k = cmInvalidIdx;
  342. // find the trk which has the next msg (min atick time)
  343. for(i=0; i<p->trkN; ++i)
  344. if( nextTrkMsg[i]!=NULL && (k==cmInvalidIdx || nextTrkMsg[i]->atick < nextTrkMsg[k]->atick) )
  345. k = i;
  346. // no next msg was found - we're done
  347. if( k == cmInvalidIdx )
  348. break;
  349. if( fl && nextTrkMsg[k]->dtick > 0 )
  350. {
  351. fl = false;
  352. nextTrkMsg[k]->dtick = 1;
  353. nextTrkMsg[k]->atick = 1;
  354. }
  355. // store the current atick
  356. atick = nextTrkMsg[k]->atick;
  357. // advance the selected track to it's next message
  358. nextTrkMsg[k] = nextTrkMsg[k]->link;
  359. // set the selected tracks next atick time
  360. if( nextTrkMsg[k] != NULL )
  361. nextTrkMsg[k]->atick = atick + nextTrkMsg[k]->dtick;
  362. }
  363. }
  364. void _cmMidiFileSetAbsoluteTime( _cmMidiFile_t* mfp )
  365. {
  366. double microsPerQN = 60000000/120; // default tempo;
  367. double microsPerTick = microsPerQN / mfp->ticksPerQN;
  368. unsigned long long amicro = 0;
  369. unsigned i;
  370. for(i=0; i<mfp->msgN; ++i)
  371. {
  372. cmMidiTrackMsg_t* mp = mfp->msgV[i];
  373. unsigned dtick = 0;
  374. if( i > 0 )
  375. {
  376. // atick must have already been set and sorted
  377. assert( mp->atick >= mfp->msgV[i-1]->atick );
  378. dtick = mp->atick - mfp->msgV[i-1]->atick;
  379. }
  380. amicro += microsPerTick * dtick;
  381. mp->amicro = amicro;
  382. // track tempo changes
  383. if( mp->status == kMetaStId && mp->metaId == kTempoMdId )
  384. microsPerTick = mp->u.iVal / mfp->ticksPerQN;
  385. }
  386. }
  387. cmMfRC_t _cmMidiFileClose( _cmMidiFile_t* mfp )
  388. {
  389. cmMfRC_t rc = kOkMfRC;
  390. if( mfp == NULL )
  391. return rc;
  392. cmMemPtrFree(&mfp->msgV);
  393. if( cmFileIsValid( mfp->fh ) )
  394. if( cmFileClose( &mfp->fh ) != kOkFileRC )
  395. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file close failed.");
  396. if( cmLHeapIsValid( mfp->lhH ) )
  397. cmLHeapDestroy(&mfp->lhH);
  398. cmMemPtrFree(&mfp);
  399. return rc;
  400. }
  401. cmMfRC_t cmMidiFileOpen( cmCtx_t* ctx, cmMidiFileH_t* hPtr, const char* fn )
  402. {
  403. cmMfRC_t rc = kOkMfRC;
  404. _cmMidiFile_t* mfp = NULL;
  405. unsigned short trkIdx = 0;
  406. cmErr_t err;
  407. unsigned i,j;
  408. if( cmMidiFileIsValid(*hPtr) )
  409. if((rc = _cmMidiFileClose(_cmMidiFileHandleToPtr(*hPtr))) != kOkMfRC )
  410. return rc;
  411. cmErrSetup(&err,&ctx->rpt,"MIDI File");
  412. // allocate the midi file object
  413. if(( mfp = cmMemAllocZ( _cmMidiFile_t, 1)) == NULL )
  414. return rc = cmErrMsg(&err,kMemAllocFailMfRC,"MIDI file memory allocation failed.");
  415. cmErrClone(&mfp->err,&err);
  416. // allocate the linked heap
  417. if( cmLHeapIsValid( mfp->lhH = cmLHeapCreate( 1024, ctx )) == false )
  418. {
  419. rc = cmErrMsg(&err,kMemAllocFailMfRC,"MIDI heap allocation failed.");
  420. goto errLabel;
  421. }
  422. // open the file
  423. if(cmFileOpen(&mfp->fh,fn,kReadFileFl | kBinaryFileFl,mfp->err.rpt) != kOkFileRC )
  424. {
  425. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file open failed.");
  426. goto errLabel;
  427. }
  428. // read header and setup track array
  429. if(( rc = _cmMidiFileReadHdr(mfp)) != kOkMfRC )
  430. goto errLabel;
  431. while( !cmFileEof(mfp->fh) && trkIdx < mfp->trkN )
  432. {
  433. unsigned chkId=0,chkN=0;
  434. // read the chunk id
  435. if((rc = _cmMidiFileRead32(mfp,&chkId)) != kOkMfRC )
  436. goto errLabel;
  437. // read the chunk size
  438. if((rc = _cmMidiFileRead32(mfp,&chkN)) != kOkMfRC )
  439. goto errLabel;
  440. // if this is not a trk chunk then skip it
  441. if( chkId != (unsigned)'MTrk')
  442. {
  443. //if( fseek( mfp->fp, chkN, SEEK_CUR) != 0 )
  444. if( cmFileSeek(mfp->fh,kCurFileFl,chkN) != kOkFileRC )
  445. {
  446. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file seek failed.");
  447. goto errLabel;
  448. }
  449. }
  450. else
  451. {
  452. if((rc = _cmMidiFileReadTrack(mfp,trkIdx)) != kOkMfRC )
  453. goto errLabel;
  454. ++trkIdx;
  455. }
  456. }
  457. // get the total trk msg count
  458. mfp->msgN = 0;
  459. for(trkIdx=0; trkIdx<mfp->trkN; ++trkIdx)
  460. mfp->msgN += mfp->trkV[ trkIdx ].cnt;
  461. // allocate the trk msg index vector: msgV[]
  462. mfp->msgV = cmMemAllocZ(cmMidiTrackMsg_t*, mfp->msgN);
  463. mfp->nextUid = 0;
  464. // store a pointer to every trk msg in msgV[] and set 'uid'
  465. for(i=0,j=0; i<mfp->trkN; ++i)
  466. {
  467. cmMidiTrackMsg_t* m = mfp->trkV[i].base;
  468. for(; m!=NULL; m=m->link)
  469. {
  470. assert( j < mfp->msgN );
  471. mfp->msgV[j++] = m;
  472. m->uid = mfp->nextUid++;
  473. }
  474. }
  475. mfp->fn = _cmMidiFileMalloc(mfp,strlen(fn)+1);
  476. assert( mfp->fn != NULL );
  477. strcpy(mfp->fn,fn);
  478. // set the atick value in each msg
  479. _cmMidiFileSetAccumulateTicks(mfp);
  480. // sort msgV[] in ascending order on atick
  481. qsort( mfp->msgV, mfp->msgN, sizeof(cmMidiTrackMsg_t*), _cmMidiFileSortFunc );
  482. // set the amicro value in each msg
  483. _cmMidiFileSetAbsoluteTime(mfp);
  484. hPtr->h = mfp;
  485. errLabel:
  486. if( cmFileClose(&mfp->fh) != kOkFileRC )
  487. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file close failed.");
  488. if( rc != kOkMfRC )
  489. _cmMidiFileClose(mfp);
  490. return rc;
  491. }
  492. cmMfRC_t cmMidiFileClose( cmMidiFileH_t* h )
  493. {
  494. cmMfRC_t rc;
  495. if( cmMidiFileIsNull(*h) )
  496. return kOkMfRC;
  497. if((rc = _cmMidiFileClose(_cmMidiFileHandleToPtr(*h))) == kOkMfRC )
  498. return rc;
  499. h->h = NULL;
  500. return rc;
  501. }
  502. cmMfRC_t _cmMidiFileWrite8( _cmMidiFile_t* mfp, unsigned char v )
  503. {
  504. cmMfRC_t rc = kOkMfRC;
  505. if( cmFileWriteUChar(mfp->fh,&v,1) != kOkFileRC )
  506. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file byte write failed.");
  507. return rc;
  508. }
  509. cmMfRC_t _cmMidiFileWrite16( _cmMidiFile_t* mfp, unsigned short v )
  510. {
  511. cmMfRC_t rc = kOkMfRC;
  512. v = mfSwap16(v);
  513. if( cmFileWriteUShort(mfp->fh,&v,1) != kOkFileRC )
  514. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file short integer write failed.");
  515. return rc;
  516. }
  517. cmMfRC_t _cmMidiFileWrite24( _cmMidiFile_t* mfp, unsigned v )
  518. {
  519. cmMfRC_t rc = kOkMfRC;
  520. unsigned mask = 0xff0000;
  521. int i;
  522. for(i=2; i>=0; --i)
  523. {
  524. unsigned char c = (v & mask) >> (i*8);
  525. mask >>= 8;
  526. if( cmFileWriteUChar(mfp->fh,&c,1) != kOkFileRC )
  527. {
  528. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file 24 bit integer write failed.");
  529. goto errLabel;
  530. }
  531. }
  532. errLabel:
  533. return rc;
  534. }
  535. cmMfRC_t _cmMidiFileWrite32( _cmMidiFile_t* mfp, unsigned v )
  536. {
  537. cmMfRC_t rc = kOkMfRC;
  538. v = mfSwap32(v);
  539. if( cmFileWriteUInt(mfp->fh,&v,1) != kOkFileRC )
  540. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file integer write failed.");
  541. return rc;
  542. }
  543. cmMfRC_t _cmMidiFileWriteRecd( _cmMidiFile_t* mfp, const void* v, unsigned byteCnt )
  544. {
  545. cmMfRC_t rc = kOkMfRC;
  546. if( cmFileWriteChar(mfp->fh,v,byteCnt) != kOkFileRC )
  547. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file write record failed.");
  548. return rc;
  549. }
  550. cmMfRC_t _cmMidiFileWriteVarLen( _cmMidiFile_t* mfp, unsigned v )
  551. {
  552. cmMfRC_t rc = kOkMfRC;
  553. unsigned buf = v & 0x7f;
  554. while((v >>= 7) > 0 )
  555. {
  556. buf <<= 8;
  557. buf |= 0x80;
  558. buf += (v & 0x7f);
  559. }
  560. while(1)
  561. {
  562. unsigned char c = (unsigned char)(buf & 0xff);
  563. if( cmFileWriteUChar(mfp->fh,&c,1) != kOkFileRC )
  564. {
  565. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"MIDI file variable length integer write failed.");
  566. goto errLabel;
  567. }
  568. if( buf & 0x80 )
  569. buf >>= 8;
  570. else
  571. break;
  572. }
  573. errLabel:
  574. return rc;
  575. }
  576. cmMfRC_t _cmMidiFileWriteHdr( _cmMidiFile_t* mfp )
  577. {
  578. cmMfRC_t rc;
  579. unsigned fileId = 'MThd';
  580. unsigned chunkByteN = 6;
  581. // write the file id ('MThd')
  582. if((rc = _cmMidiFileWrite32(mfp,fileId)) != kOkMfRC )
  583. return rc;
  584. // write the file chunk byte count (always 6)
  585. if((rc = _cmMidiFileWrite32(mfp,chunkByteN)) != kOkMfRC )
  586. return rc;
  587. // write the MIDI file format id (0,1,2)
  588. if((rc = _cmMidiFileWrite16(mfp,mfp->fmtId)) != kOkMfRC )
  589. return rc;
  590. // write the track count
  591. if((rc = _cmMidiFileWrite16(mfp,mfp->trkN)) != kOkMfRC )
  592. return rc;
  593. unsigned short v = 0;
  594. // if the ticks per quarter note field is valid ...
  595. if( mfp->ticksPerQN )
  596. v = mfp->ticksPerQN;
  597. else
  598. {
  599. // ... otherwise the division field was given in smpte
  600. v = mfp->smpteFmtId << 8;
  601. v += mfp->smpteTicksPerFrame;
  602. }
  603. if((rc = _cmMidiFileWrite16(mfp,v)) != kOkMfRC )
  604. return rc;
  605. return rc;
  606. }
  607. cmMfRC_t _cmMidiFileWriteSysEx( _cmMidiFile_t* mfp, cmMidiTrackMsg_t* tmp )
  608. {
  609. cmMfRC_t rc = kOkMfRC;
  610. if((rc = _cmMidiFileWrite8(mfp,kSysExMdId)) != kOkMfRC )
  611. goto errLabel;
  612. if( cmFileWriteUChar(mfp->fh,tmp->u.sysExPtr,tmp->byteCnt) != kOkFileRC )
  613. rc = cmErrMsg(&mfp->err,kFileFailMfRC,"Sys-ex msg write failed.");
  614. errLabel:
  615. return rc;
  616. }
  617. cmMfRC_t _cmMidiFileWriteChannelMsg( _cmMidiFile_t* mfp, const cmMidiTrackMsg_t* tmp, cmMidiByte_t* runStatus )
  618. {
  619. cmMfRC_t rc = kOkMfRC;
  620. unsigned byteN = cmMidiStatusToByteCount(tmp->status);
  621. cmMidiByte_t status = tmp->status + tmp->u.chMsgPtr->ch;
  622. if( status != *runStatus )
  623. {
  624. *runStatus = status;
  625. if((rc = _cmMidiFileWrite8(mfp,status)) != kOkMfRC )
  626. goto errLabel;
  627. }
  628. if(byteN>=1)
  629. if((rc = _cmMidiFileWrite8(mfp,tmp->u.chMsgPtr->d0)) != kOkMfRC )
  630. goto errLabel;
  631. if(byteN>=2)
  632. if((rc = _cmMidiFileWrite8(mfp,tmp->u.chMsgPtr->d1)) != kOkMfRC )
  633. goto errLabel;
  634. errLabel:
  635. return rc;
  636. }
  637. cmMfRC_t _cmMidiFileWriteMetaMsg( _cmMidiFile_t* mfp, const cmMidiTrackMsg_t* tmp )
  638. {
  639. cmMfRC_t rc;
  640. if((rc = _cmMidiFileWrite8(mfp,kMetaStId)) != kOkMfRC )
  641. return rc;
  642. if((rc = _cmMidiFileWrite8(mfp,tmp->metaId)) != kOkMfRC )
  643. return rc;
  644. switch( tmp->metaId )
  645. {
  646. case kSeqNumbMdId:
  647. if((rc = _cmMidiFileWrite8(mfp,sizeof(tmp->u.sVal))) == kOkMfRC )
  648. rc = _cmMidiFileWrite16(mfp,tmp->u.sVal);
  649. break;
  650. case kTempoMdId:
  651. if((rc = _cmMidiFileWrite8(mfp,3)) == kOkMfRC )
  652. rc = _cmMidiFileWrite24(mfp,tmp->u.iVal);
  653. break;
  654. case kSmpteMdId:
  655. if((rc = _cmMidiFileWrite8(mfp,sizeof(cmMidiSmpte_t))) == kOkMfRC )
  656. rc = _cmMidiFileWriteRecd(mfp,tmp->u.smptePtr,sizeof(cmMidiSmpte_t));
  657. break;
  658. case kTimeSigMdId:
  659. if((rc = _cmMidiFileWrite8(mfp,sizeof(cmMidiTimeSig_t))) == kOkMfRC )
  660. rc = _cmMidiFileWriteRecd(mfp,tmp->u.timeSigPtr,sizeof(cmMidiTimeSig_t));
  661. break;
  662. case kKeySigMdId:
  663. if((rc = _cmMidiFileWrite8(mfp,sizeof(cmMidiKeySig_t))) == kOkMfRC )
  664. rc = _cmMidiFileWriteRecd(mfp,tmp->u.keySigPtr,sizeof(cmMidiKeySig_t));
  665. break;
  666. case kSeqSpecMdId:
  667. if((rc = _cmMidiFileWriteVarLen(mfp,sizeof(tmp->byteCnt))) == kOkMfRC )
  668. rc = _cmMidiFileWriteRecd(mfp,tmp->u.sysExPtr,tmp->byteCnt);
  669. break;
  670. case kMidiChMdId:
  671. if((rc = _cmMidiFileWrite8(mfp,sizeof(tmp->u.bVal))) == kOkMfRC )
  672. rc = _cmMidiFileWrite8(mfp,tmp->u.bVal);
  673. break;
  674. case kEndOfTrkMdId:
  675. rc = _cmMidiFileWrite8(mfp,0);
  676. break;
  677. case kTextMdId:
  678. case kCopyMdId:
  679. case kTrkNameMdId:
  680. case kInstrNameMdId:
  681. case kLyricsMdId:
  682. case kMarkerMdId:
  683. case kCuePointMdId:
  684. {
  685. unsigned n = tmp->u.text==NULL ? 0 : strlen(tmp->u.text);
  686. if((rc = _cmMidiFileWriteVarLen(mfp,n)) == kOkMfRC && n>0 )
  687. rc = _cmMidiFileWriteRecd(mfp,tmp->u.text,n);
  688. }
  689. break;
  690. default:
  691. {
  692. // ignore unknown meta messages
  693. }
  694. }
  695. return rc;
  696. }
  697. cmMfRC_t _cmMidiFileWriteTrack( _cmMidiFile_t* mfp, unsigned trkIdx )
  698. {
  699. cmMfRC_t rc = kOkMfRC;
  700. cmMidiTrackMsg_t* tmp = mfp->trkV[trkIdx].base;
  701. cmMidiByte_t runStatus = 0;
  702. for(; tmp!=NULL; tmp=tmp->link)
  703. {
  704. // write the msg tick count
  705. if((rc = _cmMidiFileWriteVarLen(mfp,tmp->dtick)) != kOkMfRC )
  706. return rc;
  707. // switch on status
  708. switch( tmp->status )
  709. {
  710. // handle sys-ex msg
  711. case kSysExMdId:
  712. rc = _cmMidiFileWriteSysEx(mfp,tmp);
  713. break;
  714. // handle meta msg
  715. case kMetaStId:
  716. rc = _cmMidiFileWriteMetaMsg(mfp,tmp);
  717. break;
  718. default:
  719. // handle channel msg
  720. rc = _cmMidiFileWriteChannelMsg(mfp,tmp,&runStatus);
  721. }
  722. }
  723. return rc;
  724. }
  725. cmMfRC_t cmMidiFileWrite( cmMidiFileH_t h, const char* fn )
  726. {
  727. cmMfRC_t rc = kOkMfRC;
  728. _cmMidiFile_t* mfp = _cmMidiFileHandleToPtr(h);
  729. unsigned i;
  730. // create the output file
  731. if( cmFileOpen(&mfp->fh,fn,kWriteFileFl,mfp->err.rpt) != kOkFileRC )
  732. return cmErrMsg(&mfp->err,kFileFailMfRC,"The MIDI file '%s' could not be created.",cmStringNullGuard(fn));
  733. // write the file header
  734. if((rc = _cmMidiFileWriteHdr(mfp)) != kOkMfRC )
  735. {
  736. rc = cmErrMsg(&mfp->err,rc,"The file header write failed on the MIDI file '%s'.",cmStringNullGuard(fn));
  737. goto errLabel;
  738. }
  739. for(i=0; i < mfp->trkN; ++i )
  740. {
  741. unsigned chkId='MTrk';
  742. long offs0,offs1;
  743. // write the track chunk id ('MTrk')
  744. if((rc = _cmMidiFileWrite32(mfp,chkId)) != kOkMfRC )
  745. goto errLabel;
  746. cmFileTell(mfp->fh,&offs0);
  747. // write the track chunk size as zero
  748. if((rc = _cmMidiFileWrite32(mfp,0)) != kOkMfRC )
  749. goto errLabel;
  750. if((rc = _cmMidiFileWriteTrack(mfp,i)) != kOkMfRC )
  751. goto errLabel;
  752. cmFileTell(mfp->fh,&offs1);
  753. cmFileSeek(mfp->fh,kBeginFileFl,offs0);
  754. _cmMidiFileWrite32(mfp,offs1-offs0-4);
  755. cmFileSeek(mfp->fh,kBeginFileFl,offs1);
  756. }
  757. errLabel:
  758. cmFileClose(&mfp->fh);
  759. return rc;
  760. }
  761. bool cmMidiFileIsValid( cmMidiFileH_t h )
  762. { return !cmMidiFileIsNull(h); }
  763. unsigned cmMidiFileTrackCount( cmMidiFileH_t h )
  764. {
  765. _cmMidiFile_t* mfp;
  766. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  767. return cmInvalidCnt;
  768. return mfp->trkN;
  769. }
  770. unsigned cmMidiFileType( cmMidiFileH_t h )
  771. {
  772. _cmMidiFile_t* mfp;
  773. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  774. return cmInvalidId;
  775. return mfp->fmtId;
  776. }
  777. const char* cmMidiFileName( cmMidiFileH_t h )
  778. {
  779. _cmMidiFile_t* mfp;
  780. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  781. return NULL;
  782. return mfp->fn;
  783. }
  784. unsigned cmMidiFileTicksPerQN( cmMidiFileH_t h )
  785. {
  786. _cmMidiFile_t* mfp;
  787. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  788. return cmInvalidCnt;
  789. return mfp->ticksPerQN;
  790. }
  791. cmMidiByte_t cmMidiFileTicksPerSmpteFrame( cmMidiFileH_t h )
  792. {
  793. _cmMidiFile_t* mfp;
  794. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  795. return kInvalidMidiByte;
  796. if( mfp->ticksPerQN != 0 )
  797. return 0;
  798. return mfp->smpteTicksPerFrame;
  799. }
  800. cmMidiByte_t cmMidiFileSmpteFormatId( cmMidiFileH_t h )
  801. {
  802. _cmMidiFile_t* mfp;
  803. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  804. return kInvalidMidiByte;
  805. if( mfp->ticksPerQN != 0 )
  806. return 0;
  807. return mfp->smpteFmtId;
  808. }
  809. unsigned cmMidiFileTrackMsgCount( cmMidiFileH_t h, unsigned trackIdx )
  810. {
  811. _cmMidiFile_t* mfp;
  812. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  813. return cmInvalidCnt;
  814. return mfp->trkV[trackIdx].cnt;
  815. }
  816. const cmMidiTrackMsg_t* cmMidiFileTrackMsg( cmMidiFileH_t h, unsigned trackIdx )
  817. {
  818. _cmMidiFile_t* mfp;
  819. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  820. return NULL;
  821. return mfp->trkV[trackIdx].base;
  822. }
  823. unsigned cmMidiFileMsgCount( cmMidiFileH_t h )
  824. {
  825. _cmMidiFile_t* mfp;
  826. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  827. return cmInvalidCnt;
  828. return mfp->msgN;
  829. }
  830. const cmMidiTrackMsg_t** cmMidiFileMsgArray( cmMidiFileH_t h )
  831. {
  832. _cmMidiFile_t* mfp;
  833. if((mfp = _cmMidiFileHandleToPtr(h)) == NULL )
  834. return NULL;
  835. // this cast is needed to eliminate an apparently needless 'incompatible type' warning
  836. return (const cmMidiTrackMsg_t**)mfp->msgV;
  837. }
  838. unsigned cmMidiFileSeekUsecs( cmMidiFileH_t h, unsigned long long offsUSecs, unsigned* msgUsecsPtr, unsigned* microsPerTickPtr )
  839. {
  840. _cmMidiFile_t* p;
  841. if((p = _cmMidiFileHandleToPtr(h)) == NULL )
  842. return cmInvalidIdx;
  843. if( p->msgN == 0 )
  844. return cmInvalidIdx;
  845. unsigned mi;
  846. double microsPerQN = 60000000.0/120.0;
  847. double microsPerTick = microsPerQN / p->ticksPerQN;
  848. double accUSecs = 0;
  849. for(mi=0; mi<p->msgN; ++mi)
  850. {
  851. const cmMidiTrackMsg_t* mp = p->msgV[mi];
  852. if( mp->amicro >= offsUSecs )
  853. break;
  854. }
  855. if( mi == p->msgN )
  856. return cmInvalidIdx;
  857. if( msgUsecsPtr != NULL )
  858. *msgUsecsPtr = round(accUSecs - offsUSecs);
  859. if( microsPerTickPtr != NULL )
  860. *microsPerTickPtr = round(microsPerTick);
  861. return mi;
  862. }
  863. double cmMidiFileDurSecs( cmMidiFileH_t h )
  864. {
  865. _cmMidiFile_t* mfp = _cmMidiFileHandleToPtr(h);
  866. if( mfp->msgN == 0 )
  867. return 0;
  868. return mfp->msgV[ mfp->msgN-1 ]->amicro / 1000000.0;
  869. }
  870. typedef struct _cmMidiVoice_str
  871. {
  872. const cmMidiTrackMsg_t* mp;
  873. unsigned durMicros;
  874. bool sustainFl;
  875. struct _cmMidiVoice_str* link;
  876. } _cmMidiVoice_t;
  877. void _cmMidFileCalcNoteDurationReleaseNote( _cmMidiVoice_t** listPtrPtr, _cmMidiVoice_t* pp, _cmMidiVoice_t* vp )
  878. {
  879. assert( (pp==NULL && vp==*listPtrPtr) || pp->link==vp);
  880. // store the duration of the note into the track msg
  881. // assoc'd with the note-on msg
  882. cmMidiChMsg_t* cmp = (cmMidiChMsg_t*)vp->mp->u.chMsgPtr; // cast away const
  883. cmp->durMicros = vp->durMicros;
  884. _cmMidiVoice_t* np = vp->link;
  885. // release the voice msg
  886. cmMemFree(vp);
  887. // unlink the active voice msg
  888. if( pp == NULL )
  889. *listPtrPtr = np;
  890. else
  891. pp->link = np;
  892. }
  893. void _cmMidiFileCalcNoteDurationsAllocVoice( _cmMidiVoice_t** listPtrPtr, cmMidiTrackMsg_t* mp, bool sustainFl )
  894. {
  895. _cmMidiVoice_t* vp = cmMemAllocZ(_cmMidiVoice_t,1);
  896. vp->mp = mp;
  897. vp->sustainFl = sustainFl;
  898. vp->link = *listPtrPtr;
  899. *listPtrPtr = vp;
  900. }
  901. void cmMidiFileCalcNoteDurations2( cmMidiFileH_t h )
  902. {
  903. _cmMidiFile_t* p;
  904. if((p = _cmMidiFileHandleToPtr(h)) == NULL )
  905. return;
  906. if( p->msgN == 0 )
  907. return;
  908. unsigned mi = cmInvalidId;
  909. _cmMidiVoice_t* list = NULL; // list of active voices
  910. _cmMidiVoice_t* vp = NULL;
  911. cmMidiTrackMsg_t* sustainPedalDownMsg = NULL;
  912. bool sustainFlagV[ kMidiChCnt ];
  913. // clear the sustain pedal flag
  914. for(mi=0; mi<kMidiChCnt; ++mi)
  915. sustainFlagV[mi]=false;
  916. for(mi=0; mi<p->msgN; ++mi)
  917. {
  918. cmMidiTrackMsg_t* mp = p->msgV[mi];
  919. unsigned d_amicro = 0;
  920. if( mi > 0 )
  921. {
  922. assert( mp->amicro >= p->msgV[mi-1]->amicro );
  923. d_amicro = mp->amicro - p->msgV[mi-1]->amicro;
  924. }
  925. // update the duration of the sounding notes
  926. for(vp = list; vp!=NULL; vp=vp->link)
  927. vp->durMicros += d_amicro;
  928. // update the sustain pedal duration
  929. if( sustainPedalDownMsg != NULL )
  930. ((cmMidiChMsg_t*)(sustainPedalDownMsg->u.chMsgPtr))->durMicros += d_amicro; // cast away const
  931. //
  932. // If this is sustain pedal msg
  933. //
  934. if( mp->status==kCtlMdId && mp->u.chMsgPtr->d0 == kSustainCtlMdId )
  935. {
  936. unsigned chIdx = mp->u.chMsgPtr->ch;
  937. assert( chIdx < kMidiChCnt );
  938. // set the state of the sustain pedal flags
  939. sustainFlagV[chIdx] = mp->u.chMsgPtr->d1 >= 64;
  940. // if the pedal went down ...
  941. if( sustainFlagV[chIdx] )
  942. {
  943. if( sustainPedalDownMsg != NULL )
  944. {
  945. // TODO: the correct way to handle this is to maintain multiple sustain pedals
  946. //cmErrMsg(&p->err,kSustainPedalMfRC,"Sustain pedal down with no intervening sustain pedal up.");
  947. }
  948. else
  949. {
  950. sustainPedalDownMsg = mp;
  951. ((cmMidiChMsg_t*)(sustainPedalDownMsg->u.chMsgPtr))->durMicros = 0; // cast away const
  952. }
  953. _cmMidiFileCalcNoteDurationsAllocVoice( &list, mp, true );
  954. }
  955. else // ... if the pedal went up
  956. {
  957. sustainPedalDownMsg = NULL;
  958. // ... then release sustaining notes
  959. _cmMidiVoice_t* pp = NULL;
  960. for(vp=list; vp != NULL; )
  961. {
  962. _cmMidiVoice_t* np = vp->link;
  963. if( vp->sustainFl && (vp->mp->u.chMsgPtr->ch == chIdx) )
  964. _cmMidFileCalcNoteDurationReleaseNote(&list,pp,vp);
  965. else
  966. pp = vp;
  967. vp = np;
  968. }
  969. }
  970. }
  971. //
  972. // if this is a note-on msg
  973. //
  974. if( mp->status==kNoteOnMdId && mp->u.chMsgPtr->d1>0 )
  975. {
  976. _cmMidiFileCalcNoteDurationsAllocVoice( &list, mp, false );
  977. }
  978. else
  979. //
  980. // if this is a note-off msg
  981. //
  982. if( (mp->status==kNoteOnMdId && mp->u.chMsgPtr->d1==0) || (mp->status==kNoteOffMdId) )
  983. {
  984. _cmMidiVoice_t* pp = NULL;
  985. unsigned chIdx = mp->u.chMsgPtr->ch;
  986. assert( chIdx < kMidiChCnt );
  987. // for each active voice
  988. for(vp=list; vp!=NULL; vp=vp->link )
  989. {
  990. // if this active voice ch/pitch matches the note-off msg ch pitch
  991. if( (vp->mp->u.chMsgPtr->d0==mp->u.chMsgPtr->d0) && (vp->mp->u.chMsgPtr->ch==chIdx) )
  992. {
  993. // if the sustain pedal is down for this channel - then defer turning the note off
  994. if( sustainFlagV[chIdx] )
  995. vp->sustainFl = true;
  996. else
  997. _cmMidFileCalcNoteDurationReleaseNote(&list,pp,vp);
  998. break;
  999. }
  1000. pp = vp;
  1001. } // end for
  1002. } // end if
  1003. } // end-for
  1004. // check for hung notes
  1005. _cmMidiVoice_t* np = NULL;
  1006. vp = list;
  1007. while( vp != NULL )
  1008. {
  1009. np = vp->link;
  1010. if( cmMidiIsNoteOn(vp->mp->status) == false )
  1011. cmErrMsg(&p->err,kMissingNoteOffMfRC,"Missing note-off for note-on:%s",cmMidiToSciPitch(vp->mp->u.chMsgPtr->d0,NULL,0));
  1012. cmMemFree(vp);
  1013. vp = np;
  1014. }
  1015. }
  1016. bool _cmMidiFileCalcNoteDur( cmMidiTrackMsg_t* m0, cmMidiTrackMsg_t* m1, bool noteGateFl, bool sustainGateFl, bool sostGateFl )
  1017. {
  1018. // if the note is being kept sounding because the key is still depressed,
  1019. // the sustain pedal is down or it is being held by the sostenuto pedal ....
  1020. if( noteGateFl || sustainGateFl || sostGateFl )
  1021. return false; // ... do nothing
  1022. // calculate the duration of the sounding note
  1023. ((cmMidiChMsg_t*)m0->u.chMsgPtr)->durMicros = m1->amicro - m0->amicro;
  1024. return true;
  1025. }
  1026. void cmMidiFileCalcNoteDurations( cmMidiFileH_t h )
  1027. {
  1028. _cmMidiFile_t* p;
  1029. if((p = _cmMidiFileHandleToPtr(h)) == NULL )
  1030. return;
  1031. if( p->msgN == 0 )
  1032. return;
  1033. unsigned mi = cmInvalidId;
  1034. cmMidiTrackMsg_t* noteM[ kMidiNoteCnt * kMidiChCnt ]; // ptr to note-on or NULL if the note is not sounding
  1035. bool noteGateM[ kMidiNoteCnt * kMidiChCnt ]; // true if the associated note key is depressed
  1036. bool sostGateM[ kMidiNoteCnt * kMidiChCnt ]; // true if the associated note was active when the sost. pedal went down
  1037. bool sustV[kMidiChCnt]; // true if the associated sustain pedal is down
  1038. bool sostV[kMidiChCnt]; // true if the associated sostenuto pedal is down
  1039. unsigned i,j;
  1040. // initialize the state tracking variables
  1041. for(i=0; i<kMidiChCnt; ++i)
  1042. {
  1043. sustV[i] = false;
  1044. sostV[i] = false;
  1045. for(j=0; j<kMidiNoteCnt; ++j)
  1046. {
  1047. noteM[ i*kMidiNoteCnt + j ] = NULL;
  1048. noteGateM[ i*kMidiNoteCnt + j ] = false;
  1049. sostGateM[ i*kMidiNoteCnt + j ] = false;
  1050. }
  1051. }
  1052. // for each midi event
  1053. for(mi=0; mi<p->msgN; ++mi)
  1054. {
  1055. cmMidiTrackMsg_t* m = p->msgV[mi];
  1056. // verify that time is also incrementing
  1057. assert( mi==0 || (mi>0 && m->amicro >= p->msgV[mi-1]->amicro) );
  1058. // ignore all non-channel messages
  1059. if( !cmMidiIsChStatus( m->status ) )
  1060. continue;
  1061. cmMidiByte_t ch = m->u.chMsgPtr->ch; // get the midi msg channel
  1062. cmMidiByte_t d0 = m->u.chMsgPtr->d0; // get the midi msg data value
  1063. // if this is a note-on msg
  1064. if( cmMidiFileIsNoteOn(m) )
  1065. {
  1066. unsigned k = ch*kMidiNoteCnt + d0;
  1067. // a key was pressed - so it should not be currently down
  1068. if( noteGateM[k] )
  1069. cmErrWarnMsg(&p->err,kMissingNoteOffMfRC,"Missing note-off for note-on:%s",cmMidiToSciPitch(d0,NULL,0));
  1070. // there should be no existing sounding note instance for this pitch
  1071. if( noteM[k] != NULL )
  1072. cmErrWarnMsg(&p->err,kMissingNoteOffMfRC,"Missing note-off instance for note on:%s",cmMidiToSciPitch(d0,NULL,0));
  1073. noteM[k] = m;
  1074. noteGateM[k] = true;
  1075. }
  1076. else
  1077. // if this is a note-off msg
  1078. if( cmMidiFileIsNoteOff(m) )
  1079. {
  1080. unsigned k = ch*kMidiNoteCnt + d0;
  1081. cmMidiTrackMsg_t* m0 = noteM[k];
  1082. if( m0 == NULL )
  1083. cmErrWarnMsg(&p->err,kMissingNoteOffMfRC,"Missing note-on instance for note-off:%s",cmMidiToSciPitch(d0,NULL,0));
  1084. else
  1085. {
  1086. // a key was released - so it should not already be up
  1087. if( noteGateM[k]==false )
  1088. cmErrWarnMsg(&p->err,kMissingNoteOffMfRC,"Missing note-on for note-off:%s",cmMidiToSciPitch(d0,NULL,0));
  1089. noteGateM[k] = false; // update the note gate state
  1090. // update the sounding note status
  1091. if( _cmMidiFileCalcNoteDur(m0, m, noteGateM[k], sustV[ch], sostGateM[k]) )
  1092. noteM[k] = NULL;
  1093. }
  1094. }
  1095. else
  1096. // This is a sustain-pedal down msg
  1097. if( cmMidiFileIsSustainPedalDown(m) )
  1098. {
  1099. // if the sustain channel is already down
  1100. if( sustV[ch] )
  1101. cmErrWarnMsg(&p->err,kSustainPedalMfRC,"The sustain pedal went down twice with no intervening release.");
  1102. sustV[ch] = true;
  1103. }
  1104. else
  1105. // This is a sustain-pedal up msg
  1106. if( cmMidiFileIsSustainPedalUp(m) )
  1107. {
  1108. // if the sustain channel is already up
  1109. if( sustV[ch]==false )
  1110. cmErrWarnMsg(&p->err,kSustainPedalMfRC,"The sustain pedal release message was received with no previous pedal down.");
  1111. sustV[ch] = false;
  1112. unsigned k = ch*kMidiNoteCnt;
  1113. // for each sounding note on this channel
  1114. for(; k<ch*kMidiNoteCnt+kMidiNoteCnt; ++k)
  1115. if( noteM[k]!=NULL && _cmMidiFileCalcNoteDur(noteM[k], m, noteGateM[k], sustV[ch], sostGateM[k]) )
  1116. noteM[k] = NULL;
  1117. }
  1118. else
  1119. // This is a sostenuto-pedal down msg
  1120. if( cmMidiFileIsSostenutoPedalDown(m) )
  1121. {
  1122. // if the sustain channel is already down
  1123. if( sostV[ch] )
  1124. cmErrWarnMsg(&p->err,kSostenutoPedalMfRC,"The sostenuto pedal went down twice with no intervening release.");
  1125. // record the notes that are active when the sostenuto pedal went down
  1126. unsigned k = ch * kMidiNoteCnt;
  1127. cmVOB_Copy( sostGateM + k, kMidiNoteCnt, noteGateM + k);
  1128. sostV[ch] = true;
  1129. }
  1130. else
  1131. // This is a sostenuto-pedal up msg
  1132. if( cmMidiFileIsSostenutoPedalUp(m) )
  1133. {
  1134. // if the sustain channel is already up
  1135. if( sostV[ch]==false )
  1136. cmErrWarnMsg(&p->err,kSostenutoPedalMfRC,"The sostenuto pedal release message was received with no previous pedal down.");
  1137. sostV[ch] = false;
  1138. // for each note on this channel
  1139. unsigned k = ch*kMidiNoteCnt;
  1140. for(; k<ch*kMidiNoteCnt+kMidiNoteCnt; ++k)
  1141. {
  1142. sostGateM[k] = false;
  1143. if( noteM[k]!=NULL && _cmMidiFileCalcNoteDur(noteM[k], m, noteGateM[k], sustV[ch], sostGateM[k]) )
  1144. noteM[k] = NULL;
  1145. }
  1146. }
  1147. } // for each midi file event
  1148. }
  1149. void cmMidiFileSetDelay( cmMidiFileH_t h, unsigned ticks )
  1150. {
  1151. _cmMidiFile_t* p;
  1152. unsigned mi;
  1153. if((p = _cmMidiFileHandleToPtr(h)) == NULL )
  1154. return;
  1155. if( p->msgN == 0 )
  1156. return;
  1157. for(mi=0; mi<p->msgN; ++mi)
  1158. {
  1159. cmMidiTrackMsg_t* mp = p->msgV[mi];
  1160. // locate the first msg which has a non-zero delta tick
  1161. if( mp->dtick > 0 )
  1162. {
  1163. mp->dtick = ticks;
  1164. break;
  1165. }
  1166. }
  1167. }
  1168. unsigned cmMidiFilePackTrackMsgBufByteCount( const cmMidiTrackMsg_t* m )
  1169. { return sizeof(cmMidiTrackMsg_t) + m->byteCnt; }
  1170. cmMidiTrackMsg_t* cmMidiFilePackTrackMsg( const cmMidiTrackMsg_t* m, void* buf, unsigned bufByteCnt )
  1171. {
  1172. unsigned n = sizeof(cmMidiTrackMsg_t) + m->byteCnt;
  1173. if( n < bufByteCnt )
  1174. {
  1175. assert(0);
  1176. return NULL;
  1177. }
  1178. // copy the cmMidiTrackMsg_t into the buffer
  1179. memcpy(buf, m, sizeof(cmMidiTrackMsg_t));
  1180. if( m->byteCnt > 0 )
  1181. {
  1182. // copy any linked data into the buffer
  1183. memcpy(buf + sizeof(cmMidiTrackMsg_t), m->u.voidPtr, m->byteCnt );
  1184. // fixup the linked data ptr
  1185. cmMidiTrackMsg_t* mp = (cmMidiTrackMsg_t*)buf;
  1186. mp->u.voidPtr = buf + sizeof(cmMidiTrackMsg_t);
  1187. }
  1188. return (cmMidiTrackMsg_t*)buf;
  1189. }
  1190. void _cmMidiFilePrintHdr( const _cmMidiFile_t* mfp, cmRpt_t* rpt )
  1191. {
  1192. if( mfp->fn != NULL )
  1193. cmRptPrintf(rpt,"%s ",mfp->fn);
  1194. cmRptPrintf(rpt,"fmt:%i ticksPerQN:%i tracks:%i\n",mfp->fmtId,mfp->ticksPerQN,mfp->trkN);
  1195. cmRptPrintf(rpt," UID dtick atick amicro type ch D0 D1\n");
  1196. cmRptPrintf(rpt,"----- ---------- ---------- ---------- : ---- --- --- ---\n");
  1197. }
  1198. void _cmMidiFilePrintMsg( cmRpt_t* rpt, const cmMidiTrackMsg_t* tmp )
  1199. {
  1200. cmRptPrintf(rpt,"%5i %10u %10llu %10llu : ",
  1201. tmp->uid,
  1202. tmp->dtick,
  1203. tmp->atick,
  1204. tmp->amicro );
  1205. if( tmp->status == kMetaStId )
  1206. {
  1207. cmRptPrintf(rpt,"%s ", cmMidiMetaStatusToLabel(tmp->metaId));
  1208. }
  1209. else
  1210. {
  1211. cmRptPrintf(rpt,"%4s %3i %3i %3i",
  1212. cmMidiStatusToLabel(tmp->status),
  1213. tmp->u.chMsgPtr->ch,
  1214. tmp->u.chMsgPtr->d0,
  1215. tmp->u.chMsgPtr->d1);
  1216. }
  1217. if( cmMidiIsChStatus(tmp->status) && cmMidiIsNoteOn(tmp->status) && (tmp->u.chMsgPtr->d1>0) )
  1218. cmRptPrintf(rpt," %4s ",cmMidiToSciPitch(tmp->u.chMsgPtr->d0,NULL,0));
  1219. cmRptPrintf(rpt,"\n");
  1220. }
  1221. void cmMidiFilePrintMsgs( cmMidiFileH_t h, cmRpt_t* rpt )
  1222. {
  1223. const _cmMidiFile_t* p = _cmMidiFileHandleToPtr(h);
  1224. unsigned mi;
  1225. _cmMidiFilePrintHdr(p,rpt);
  1226. for(mi=0; mi<p->msgN; ++mi)
  1227. {
  1228. cmMidiTrackMsg_t* mp = p->msgV[mi];
  1229. if( mp != NULL )
  1230. _cmMidiFilePrintMsg(rpt,mp);
  1231. }
  1232. }
  1233. void cmMidiFilePrintTracks( cmMidiFileH_t h, unsigned trkIdx, cmRpt_t* rpt )
  1234. {
  1235. const _cmMidiFile_t* mfp = _cmMidiFileHandleToPtr(h);
  1236. _cmMidiFilePrintHdr(mfp,rpt);
  1237. int i = trkIdx == cmInvalidIdx ? 0 : trkIdx;
  1238. int n = trkIdx == cmInvalidIdx ? mfp->trkN : trkIdx+1;
  1239. for(; i<n; ++i)
  1240. {
  1241. cmRptPrintf(rpt,"Track:%i\n",i);
  1242. cmMidiTrackMsg_t* tmp = mfp->trkV[i].base;
  1243. while( tmp != NULL )
  1244. {
  1245. _cmMidiFilePrintMsg(rpt,tmp);
  1246. tmp = tmp->link;
  1247. }
  1248. }
  1249. }
  1250. bool cmMidiFileIsNull( cmMidiFileH_t h )
  1251. { return (_cmMidiFile_t*)h.h == NULL; }
  1252. void cmMidiFileTestPrint( void* printDataPtr, const char* fmt, va_list vl )
  1253. { vprintf(fmt,vl); }
  1254. cmMfRC_t cmMidiFileGenPlotFile( cmCtx_t* ctx, const cmChar_t* midiFn, const cmChar_t* outFn )
  1255. {
  1256. cmMfRC_t rc = kOkMfRC;
  1257. cmMidiFileH_t mfH = cmMidiFileNullHandle;
  1258. cmFileH_t fH = cmFileNullHandle;
  1259. unsigned i = 0;
  1260. const cmMidiTrackMsg_t** m = NULL;
  1261. unsigned mN = 0;
  1262. if((rc = cmMidiFileOpen(ctx, &mfH, midiFn )) != kOkMfRC )
  1263. return cmErrMsg(&ctx->err,rc,"The MIDI file object could not be opened from '%s'.",cmStringNullGuard(midiFn));
  1264. _cmMidiFile_t* p = _cmMidiFileHandleToPtr(mfH);
  1265. if( (m = cmMidiFileMsgArray(mfH)) == NULL || (mN = cmMidiFileMsgCount(mfH)) == 0 )
  1266. {
  1267. rc = cmErrMsg(&p->err,kFileFailMfRC,"The MIDI file object appears to be empty.");
  1268. goto errLabel;
  1269. }
  1270. cmMidiFileCalcNoteDurations( mfH );
  1271. if( cmFileOpen(&fH,outFn,kWriteFileFl,p->err.rpt) != kOkFileRC )
  1272. return cmErrMsg(&p->err,kFileFailMfRC,"Unable to create the file '%s'.",cmStringNullGuard(outFn));
  1273. for(i=0; i<mN; ++i)
  1274. if( (m[i]!=NULL) && cmMidiIsChStatus(m[i]->status) && cmMidiIsNoteOn(m[i]->status) && (m[i]->u.chMsgPtr->d1>0) )
  1275. cmFilePrintf(fH,"n %f %f %i %s\n",m[i]->amicro/1000000.0,m[i]->u.chMsgPtr->durMicros/1000000.0,m[i]->uid,cmMidiToSciPitch(m[i]->u.chMsgPtr->d0,NULL,0));
  1276. errLabel:
  1277. cmMidiFileClose(&mfH);
  1278. cmFileClose(&fH);
  1279. return rc;
  1280. }
  1281. void cmMidiFilePrintControlNumbers( cmCtx_t* ctx, const char* fn )
  1282. {
  1283. cmMidiFileH_t h = cmMidiFileNullHandle;
  1284. cmMfRC_t rc;
  1285. if((rc = cmMidiFileOpen(ctx, &h, fn )) != kOkMfRC )
  1286. {
  1287. cmErrMsg(&ctx->err,rc,"MIDI file open failed on '%s'.",fn);
  1288. goto errLabel;
  1289. }
  1290. const cmMidiTrackMsg_t** mm;
  1291. unsigned n = cmMidiFileMsgCount(h);
  1292. if((mm = cmMidiFileMsgArray(h)) != NULL )
  1293. {
  1294. unsigned j;
  1295. for(j=0; j<n; ++j)
  1296. {
  1297. const cmMidiTrackMsg_t* m = mm[j];
  1298. if( m->status == kCtlMdId && m->u.chMsgPtr->d0==66 )
  1299. printf("%i %i\n",m->u.chMsgPtr->d0,m->u.chMsgPtr->d1);
  1300. }
  1301. }
  1302. errLabel:
  1303. cmMidiFileClose(&h);
  1304. }
  1305. void cmMidiFileTest( const char* fn, cmCtx_t* ctx )
  1306. {
  1307. cmMfRC_t rc;
  1308. cmMidiFileH_t h = cmMidiFileNullHandle;
  1309. if((rc = cmMidiFileOpen(ctx,&h,fn)) != kOkMfRC )
  1310. {
  1311. printf("Error:%i Unable to open the cmMidi file: %s\n",rc,fn);
  1312. return;
  1313. }
  1314. //cmMidiFileCalcNoteDurations( h );
  1315. if( 1 )
  1316. {
  1317. //cmMidiFileTickToMicros( h );
  1318. //cmMidiFileTickToSamples(h,96000,false);
  1319. cmMidiFilePrintMsgs(h,&ctx->rpt);
  1320. }
  1321. if( 0 )
  1322. {
  1323. //cmMidiFilePrint(h,cmMidiFileTrackCount(h)-1,&ctx->rpt);
  1324. //cmMidiFilePrint(h,cmInvalidIdx,&ctx->rpt);
  1325. cmMidiFilePrintControlNumbers(ctx, fn );
  1326. }
  1327. if( 0 )
  1328. {
  1329. printf("Tracks:%i\n",cmMidiFileTrackCount(h));
  1330. unsigned i = 0;
  1331. for(i=0; i<cmMidiFileMsgCount(h); ++i)
  1332. {
  1333. cmMidiTrackMsg_t* tmp = (cmMidiTrackMsg_t*)cmMidiFileMsgArray(h)[i];
  1334. if( tmp->status==kMetaStId && tmp->metaId == kTempoMdId )
  1335. {
  1336. double bpm = 60000000.0/tmp->u.iVal;
  1337. printf("Tempo:%i %f\n",tmp->u.iVal,bpm);
  1338. tmp->u.iVal = floor( 60000000.0/69.0 );
  1339. break;
  1340. }
  1341. }
  1342. cmMidiFileWrite(h,"/home/kevin/temp/test0.mid");
  1343. }
  1344. cmMidiFileClose(&h);
  1345. }