Mercurial > ~darius > hgwebdir.cgi > mikmod
view playercode/mloader.c @ 4:5d614bcc4287
Initial entry of mikmod into the CVS tree.
author | darius |
---|---|
date | Fri, 23 Jan 1998 16:05:08 +0000 |
parents | |
children |
line wrap: on
line source
/* Name: MLOADER.C Description: These routines are used to access the available module loaders Portability: All systems - all compilers */ #include <string.h> #include "mikmod.h" FILE *modfp; UNIMOD of; static MLOADER *firstloader = NULL; UWORD finetune[16] = { 8363, 8413, 8463, 8529, 8581, 8651, 8723, 8757, 7895, 7941, 7985, 8046, 8107, 8169, 8232, 8280 }; void ML_InfoLoader(void) { int t; MLOADER *l; // list all registered devicedrivers: for(t=1,l=firstloader; l!=NULL; l=l->next, t++) printf("%d. %s\n",t,l->version); } void ML_RegisterLoader(MLOADER *ldr) { MLOADER *cruise = firstloader; if(cruise!=NULL) { while(cruise->next!=NULL) cruise = cruise->next; cruise->next = ldr; } else firstloader = ldr; } BOOL ReadComment(UWORD len) { //int t; if(len) { if(!(of.comment=(CHAR *)_mm_malloc(len+1))) return 0; fread(of.comment,len,1,modfp); of.comment[len] = 0; } return 1; } BOOL AllocPositions(int total) { if((of.positions = _mm_calloc(total,sizeof(UWORD))) == NULL) return 0; return 1; } BOOL AllocPatterns(void) { int s,t,tracks = 0; // Allocate track sequencing array if(!(of.patterns = (UWORD *)_mm_calloc((ULONG)(of.numpat+1)*of.numchn,sizeof(UWORD)))) return 0; if(!(of.pattrows = (UWORD *)_mm_calloc(of.numpat+1,sizeof(UWORD)))) return 0; for(t=0; t<of.numpat+1; t++) { of.pattrows[t] = 64; for(s=0; s<of.numchn; s++) of.patterns[(t*of.numchn)+s] = tracks++; } return 1; } BOOL AllocTracks(void) { if(!(of.tracks=(UBYTE **)_mm_calloc(of.numtrk,sizeof(UBYTE *)))) return 0; return 1; } BOOL AllocInstruments(void) { int t,n; if((of.instruments=(INSTRUMENT *)_mm_calloc(of.numins,sizeof(INSTRUMENT)))==NULL) return 0; for(t=0; t<of.numins; t++) { for(n=0; n<120; n++) // Init note / sample lookup table { of.instruments[t].samplenote[n] = n; of.instruments[t].samplenumber[n] = t; } of.instruments[t].globvol = 64; } return 1; } BOOL AllocSamples(void) { UWORD u; if((of.samples = (SAMPLE *)_mm_calloc(of.numsmp,sizeof(SAMPLE)))==NULL) return 0; for(u=0; u<of.numsmp; u++) { of.samples[u].panning = 128; of.samples[u].handle = -1; of.samples[u].globvol = 64; of.samples[u].volume = 64; } return 1; } BOOL ML_LoadSamples(void) { SAMPLE *s; int u; for(u=of.numsmp, s=of.samples; u; u--, s++) if(s->length) SL_RegisterSample(s,MD_MUSIC,modfp); return 1; } CHAR *DupStr(CHAR *s, UWORD len) // Creates a CSTR out of a character buffer of 'len' bytes, but strips // any terminating non-printing characters like 0, spaces etc. { UWORD t; CHAR *d = NULL; // Scan for first printing char in buffer [includes high ascii up to 254] while(len) { if(s[len-1] > 0x20) break; len--; } // When the buffer wasn't completely empty, allocate // a cstring and copy the buffer into that string, except // for any control-chars #ifdef __GNUC__ if(len<16) len = 16; #endif if((d=(CHAR *)_mm_malloc(len+1)) != NULL) { for(t=0; t<len; t++) d[t] = (s[t]<32) ? ' ' : s[t]; d[t] = 0; } return d; } static void ML_XFreeSample(SAMPLE *s) { if(s->handle>=0) MD_SampleUnLoad(s->handle); if(s->samplename!=NULL) free(s->samplename); } static void ML_XFreeInstrument(INSTRUMENT *i) { if(i->insname!=NULL) free(i->insname); } static void ML_FreeEx(UNIMOD *mf) { UWORD t; if(mf->songname!=NULL) free(mf->songname); if(mf->composer!=NULL) free(mf->composer); if(mf->comment!=NULL) free(mf->comment); if(mf->modtype!=NULL) free(mf->modtype); if(mf->positions!=NULL) free(mf->positions); if(mf->patterns!=NULL) free(mf->patterns); if(mf->pattrows!=NULL) free(mf->pattrows); if(mf->tracks!=NULL) { for(t=0; t<mf->numtrk; t++) if(mf->tracks[t]!=NULL) free(mf->tracks[t]); free(mf->tracks); } if(mf->instruments != NULL) { for(t=0; t<mf->numins; t++) ML_XFreeInstrument(&mf->instruments[t]); free(mf->instruments); } if(mf->samples != NULL) { for(t=0; t<mf->numsmp; t++) if(mf->samples[t].length) ML_XFreeSample(&mf->samples[t]); free(mf->samples); } memset(mf,0,sizeof(UNIMOD)); } static UNIMOD *ML_AllocUniMod(void) { UNIMOD *mf; if((mf=_mm_calloc(1,sizeof(UNIMOD))) == NULL) return NULL; return mf; } /****************************************** Next are the user-callable functions ******************************************/ void MikMod_FreeSong(UNIMOD *mf) { if(mf!=NULL) { Player_Exit(mf); ML_FreeEx(mf); } } CHAR *MikMod_LoadSongTitle(CHAR *filename) { MLOADER *l; CHAR *retval; FILE *fp; if((fp = _mm_fopen(filename,"rb"))==NULL) return NULL; _mm_errno = 0; _mm_critical = 0; _mm_iobase_setcur(modfp); // Try to find a loader that recognizes the module for(l=firstloader; l!=NULL; l=l->next) { _mm_rewind(modfp); if(l->Test()) break; } if(l==NULL) { _mm_errno = MMERR_NOT_A_MODULE; _mm_iobase_revert(); if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } retval = l->LoadTitle(); fclose(fp); return(retval); } UNIMOD *MikMod_LoadSongFP(FILE *fp, int maxchan) // Loads a module given a file pointer. // File is loaded from the current file seek position. { int t; MLOADER *l; BOOL ok; UNIMOD *mf; modfp = fp; _mm_errno = 0; _mm_critical = 0; _mm_iobase_setcur(modfp); // Try to find a loader that recognizes the module for(l=firstloader; l!=NULL; l=l->next) { _mm_rewind(modfp); if(l->Test()) break; } if(l==NULL) { _mm_errno = MMERR_NOT_A_MODULE; _mm_iobase_revert(); if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } // init unitrk routines if(!UniInit()) { if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } // load the song using the song's loader variable memset(&of,0,sizeof(UNIMOD)); of.initvolume = 128; // init panning array for(t=0; t<64; t++) of.panning[t] = ((t+1)&2) ? 255 : 0; for(t=0; t<64; t++) of.chanvol[t] = 64; // init module loader and load the header / patterns if(l->Init()) { _mm_rewind(modfp); ok = l->Load(); } else ok = 0; // free loader and unitrk allocations l->Cleanup(); UniCleanup(); if(!ok) { ML_FreeEx(&of); _mm_iobase_revert(); if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } if(!ML_LoadSamples()) { ML_FreeEx(&of); _mm_iobase_revert(); if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } if((mf=ML_AllocUniMod()) == NULL) { ML_FreeEx(&of); _mm_iobase_revert(); if(_mm_errorhandler!=NULL) _mm_errorhandler(); return NULL; } // Copy the static UNIMOD contents into the dynamic UNIMOD struct. memcpy(mf,&of,sizeof(UNIMOD)); _mm_iobase_revert(); if(maxchan > 0) { if(!(mf->flags & UF_NNA) && (mf->numchn < maxchan)) maxchan = mf->numchn; else if((mf->numvoices!=0) && mf->numvoices < maxchan) maxchan = mf->numvoices; if(maxchan < mf->numchn) mf->flags |= UF_NNA; if(MikMod_SetNumVoices(maxchan,-1)) { MikMod_FreeSong(mf); return NULL; } } return mf; } UNIMOD *MikMod_LoadSong(CHAR *filename, int maxchan) // Open a module via it's filename. The loader will initialize the specified // song-player 'player'. { FILE *fp; UNIMOD *mf; if((fp = _mm_fopen(filename,"rb"))==NULL) return NULL; if((mf = MikMod_LoadSongFP(fp, maxchan)) != NULL) { if(SL_LoadSamples() || Player_Init(mf)) { MikMod_FreeSong(mf); mf = NULL; } } fclose(fp); return mf; }