changeset 4:5d614bcc4287

Initial entry of mikmod into the CVS tree.
author darius
date Fri, 23 Jan 1998 16:05:08 +0000 (1998-01-23)
parents 71e20a32bd84
children 42e11dc15457
files docs/bugs.txt docs/future.txt docs/mikcvt.txt docs/mikmod.doc docs/tips.txt playercode/drv_nos.c playercode/drv_wav.c playercode/load_669.c playercode/load_dsm.c playercode/load_far.c playercode/load_it.c playercode/load_m15.c playercode/load_med.c playercode/load_mod.c playercode/load_mtm.c playercode/load_s3m.c playercode/load_stm.c playercode/load_ult.c playercode/load_uni.c playercode/load_wav.c playercode/load_xm.c playercode/mloader.c playercode/stream.c playercode/virtch2.c
diffstat 24 files changed, 7301 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/docs/bugs.txt	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,25 @@
+
+ --> MikMod Sound Libraries Version 3.0
+  -> Known Bugs - August 20st, 1997
+
+
+ If anyone knows of a fix for any of the bugs listed below, please contact
+ me at dracoirs@epix.net and tell me about it.
+
+ - The DSM loader has been changed from MikMod 2.10 to 3.0 and is un-
+   tested.  It may not work.
+ 
+ - 669 loader is far from perfect.  Thanks go out to the complete lack of
+   useful docs for this format.  It seems to play most 669's adequately,
+   however.
+
+ - Since there is no 100% sure detection of UST / Soundtracker the 15-
+   instrument loader may, under very rare circumstances, detect a sound-
+   tracker module as UST and play it incorrectly.
+
+ - MikMod locks Aztec soundcards (SB clones distributed with Packard Bell
+   computers).  MikMod will work, but it hard locks the card and the
+   user must turn off the computer to reset it (Windows can't even do it!)
+
+   This bug may not exist anymore; someone tell me if it does.
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/docs/future.txt	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,26 @@
+
+ --> MikMod Sound Libraries Version 3.0
+  -> Changes Planned for the Future - August 20th, 1997
+
+
+ - Optional Interpolated mixing in VIRTCH.C [real-time].
+
+ - Floating point alternative to 64-bit ints in VIRTCH2.C.
+
+ - Support for stereo samples and sound effects.
+
+ - Audio streaming from disk for large raw audio files (WAV, etc).
+
+ - Native Gravis Ultrasound Plug and Play soundcard support.
+
+ - Better AWE32 support (AWE32 does not support software sound-fx
+   mixing).
+
+ - Native AWE64 support [does anyone have GOOD specs on this or the
+   AWE32?]
+
+ - Windows 95 and DirectSound [ugh] Drivers.
+
+ - Things people suggest - send suggestions to dracoirs@epix.net and
+   don't forget to visit the webpage at http://www.epix.net/~dracoirs/
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/docs/mikcvt.txt	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,76 @@
+
+ --> MikMod Sound System  v3.0
+  -> UniFormat Conversion Program (MIKCVT.EXE)
+
+
+
+-/- Introduction
+================
+
+ MikMod utilizes its own internal format called UniMod (or UniFormat).
+ This internal format can 100% accurately represent any of the modules
+ that MikMod supports.  The advantage of UniMod is that the programmer
+ can convert all modules of any type to this native MikMod format, and
+ simply include the UniMod loader alone into the program.
+
+ For more information about UniMod, see UNIMOD30.TXT.
+
+ MikMod comes with a program called MIKCVT.EXE, which converts modules
+ to the UniMod format.  MIKCVT will load any module supported by
+ MikMod, optimize and compress the pattern data, and save out a file
+ with the .UNI extension.  MIKCVT also supports wildcards, so that you
+ can easily convert several modules at a time.
+
+
+-/- Distribution
+================
+
+ MIKCVT is a totally free utility, and can be downloaded separately
+ from MikMod.  MIKCVT comes with full source code, and the programmer
+ can feel free to modify this source code in any way for PERSONAL use
+ only (to rearrange the format to prevent module ripping for instance).
+ PLEASE DO NOT REDISTRIBUTE A MODIFIED VERSION OF MIKCVT.  The last
+ thing anyone wants is to have odd non-std versions of UniMod modules
+ floating around.
+
+
+
+-/- Usage
+=========
+
+ MIKCVT has a very simple command line interface, as follows:
+
+   MIKCVT  [-strip] infile [infile] [infile] ...
+
+   MIKCVT  song1.s3m song2.xm
+   MIKCVT  -strip *.s3m
+   MIKCVT  *.*
+
+ The first example shows the general syntax of MIKCVT.  Note that
+ -strip is optional, but if present MUST be the first option on
+ the command line.
+
+ In any instance, the -strip is optional, and if present will tell
+ MIKCVT to remove all text strings from the module.  This includes
+ the title, sample names, instrument names, comments and other forms
+ of text.  This feature is generally intended for use by game and demo
+ developers, who may not require such information in the release pro-
+ duct.
+
+ 
+-/- Miscellaneous
+=================
+
+ There will probably be a future version of MIKCVT that will further
+ optimize modules in -strip mode.
+
+ For information, questions, comments, or various chit-chat, contact
+ Jake Stine / Divine Entertainment at jstine@hotmail.com.
+
+ For updates and other updated information, check out the Divine Ent-
+ ertainment website at http://www.epix.net/~dracoirs/
+
+
+ That's all for today.  Have fun!
+
+
Binary file docs/mikmod.doc has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/docs/tips.txt	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,64 @@
+
+ --> MikMod 3.0
+  -> Various Programming Tips and Tricks!
+
+
+ Here are some tips and tricks that I have jotted down as I think of them.
+ The list is short for now, but it will no doubt grow as more are thought
+ up.  Also, check for additional information on the Divine Entertainment
+ website at http://www.epix.net/~dracoirs/
+
+ Ok, so this little text file is pretty short right now.  These things
+ take time to think of.. :-)
+
+
+-/- Compiling / Makefiles
+
+ Watcom:
+
+   The example files that come with MikMod default to 30k stack buffers
+   (option stack=30000 on the wlink command line).  This is a safety pre-
+   caution and by no means a requirement.
+
+   You can usually safely use a 10k stack if you are using an interrupt
+   driven MikMod_Update() and you can safely use a 5k stack if you are
+   using polling (calling MikMod_Update() from a main program loop).
+
+                               ----------
+
+   If you are writing a module player and are NOT calling MikMod_Update()
+   from an interrupt, then you can safely remove /ZU from the following
+   modules in the /MikMod directory:
+
+      MDRIVER.C
+      VIRTCH.C
+
+   Removing /ZU will slightly reduce code size and increase replay speed.
+
+                               ----------
+   
+   If you are using the IDE and want to add PMODE/W to the list of valid
+   targets, follow these steps:
+
+     a) copy PMWBIND.EXE and PMODEW.LNK to your watcom/binw diectory.
+
+     b) Load WSYSTEM.LNK, and inster the contents of PMODEW.LINK into it
+        at the very top.
+
+     c) Load IDEDOS32.CFG.  Insert a line that reads:
+            Target *.exe, dw2e_, "PMODE/W Executable"
+
+     d) Load IDE.CFG.  Around line 840, you will find the following line:
+          VSwitch  0, dr2??, "System:", SYS, " ", ONE, REQ, dos4g
+        Inser this line below it:
+          VSwitch  0, dw2??, "System:", SYS, " ", ONE, REQ, pmodew
+
+     e) This is the tricky part.  There are several lines in IDE.CFG that
+        contain 'dr2'.  Each of these [excluding the special case in step
+        d] must be duplicated, and 'dr2' replaced with 'dw2' (no other
+        changes required).  Do this right and you will have full IDE cap-
+        ability with the pmode/w extender!
+
+   There you go, all set to compile with pmode/w within the Watcom IDE!
+
+                               ----------
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/drv_nos.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,160 @@
+/*
+
+Name:
+DRV_NOS.C
+
+Description:
+Mikmod driver for no output on any soundcard, monitor, keyboard, or whatever :)
+
+Portability:
+All systems - All compilers
+
+*/
+
+#include "mikmod.h"
+
+
+static BOOL NS_IsThere(void)
+{
+    return 1;
+}
+
+
+static SWORD NS_SampleLoad(SAMPLOAD *s, int type, FILE *fp)
+{
+    return 0;
+}
+
+
+static void NS_SampleUnload(SWORD h)
+{
+}
+
+
+static ULONG NS_SampleSpace(int type)
+{
+    return 0;
+}
+
+
+static ULONG NS_SampleLength(int type, SAMPLE *s)
+{
+    return s->length;
+}
+
+
+static BOOL NS_Init(void)
+{
+    return 0;
+}
+
+
+static void NS_Exit(void)
+{
+}
+
+
+static BOOL NS_Reset(void)
+{
+    return 0;
+}
+
+
+static BOOL NS_PlayStart(void)
+{
+}
+
+
+static void NS_PlayStop(void)
+{
+}
+
+
+static void NS_Update(void)
+{
+}
+
+
+static BOOL NS_SetNumVoices(void)
+{
+    return 0;
+}
+
+
+static void NS_VoiceSetVolume(UBYTE voice,UWORD vol)
+{
+}
+
+
+static void NS_VoiceSetFrequency(UBYTE voice,ULONG frq)
+{
+}
+
+
+static void NS_VoiceSetPanning(UBYTE voice,ULONG pan)
+{
+}
+
+
+static void NS_VoicePlay(UBYTE voice,SWORD handle,ULONG start,ULONG size,ULONG reppos,ULONG repend,UWORD flags)
+{
+}
+
+
+static void NS_VoiceStop(UBYTE voice)
+{
+}
+
+
+static BOOL NS_VoiceStopped(UBYTE voice)
+{
+   return 0;
+}
+
+
+static void NS_VoiceReleaseSustain(UBYTE voice)
+{
+}
+
+
+static SLONG NS_VoiceGetPosition(UBYTE voice)
+{
+   return 0;
+}
+
+
+static ULONG NS_VoiceRealVolume(UBYTE voice)
+{
+   return 0;
+}
+
+
+
+MDRIVER drv_nos =
+{   NULL,
+    "No Sound",
+    "Nosound Driver v2.0 - (c) Creative Silence",
+    255,255,
+    NS_IsThere,
+    NS_SampleLoad,
+    NS_SampleUnload,
+    NS_SampleSpace,
+    NS_SampleLength,
+    NS_Init,
+    NS_Exit,
+    NS_Reset,
+    NS_SetNumVoices,
+    NS_PlayStart,
+    NS_PlayStop,
+    NS_Update,
+    NS_VoiceSetVolume,
+    NS_VoiceSetFrequency,
+    NS_VoiceSetPanning,
+    NS_VoicePlay,
+    NS_VoiceStop,
+    NS_VoiceStopped,
+    NS_VoiceReleaseSustain,
+    NS_VoiceGetPosition,
+    NS_VoiceRealVolume
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/drv_wav.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,116 @@
+
+#include "mikmod.h"
+
+#ifdef __GNUC__
+#include <sys/types.h>
+#else
+#include <io.h>
+#endif
+#include <sys/stat.h>
+#include <fcntl.h>
+
+#define WAVBUFFERSIZE 65536
+
+static FILE *wavout;
+
+static SBYTE *WAV_DMABUF;
+static ULONG dumpsize;
+
+static BOOL WAV_IsThere(void)
+{
+    return 1;
+}
+
+
+static BOOL WAV_Init(void)
+{
+    if(NULL == (wavout = fopen("music.wav", "wb"))) return 1;
+    if(NULL == (WAV_DMABUF = _mm_malloc(WAVBUFFERSIZE))) return 1;
+
+    md_mode |= DMODE_SOFT_MUSIC | DMODE_SOFT_SNDFX;
+
+    if(VC_Init()) return 1;
+    
+    _mm_write_string("RIFF    WAVEfmt ",wavout);
+    _mm_write_I_ULONG(16,wavout);     // length of this RIFF block crap
+
+    _mm_write_I_UWORD(1, wavout);    // microsoft format type
+    _mm_write_I_UWORD((md_mode & DMODE_STEREO) ? 2 : 1, wavout);
+    _mm_write_I_ULONG(md_mixfreq, wavout);
+    _mm_write_I_ULONG(md_mixfreq * ((md_mode & DMODE_STEREO) ? 2 : 1) *
+                      ((md_mode & DMODE_16BITS) ? 2 : 1), wavout);
+
+    _mm_write_I_UWORD(((md_mode & DMODE_16BITS) ? 2 : 1) * 
+                      ((md_mode & DMODE_STEREO) ? 2 : 1), wavout);    // block alignment (8/16 bit)
+
+    _mm_write_I_UWORD((md_mode & DMODE_16BITS) ? 16 : 8,wavout);
+
+    _mm_write_string("data",wavout);
+
+    dumpsize = 0;
+
+    return 0;
+}
+
+
+static void WAV_Exit(void)
+{
+    VC_Exit();
+
+    // write in the actual sizes now
+
+    if(wavout!=NULL)
+    {   _mm_fseek(wavout,4,SEEK_SET);
+        _mm_write_I_ULONG(dumpsize + 32, wavout);
+        _mm_fseek(wavout,40,SEEK_SET);
+        _mm_write_I_ULONG(dumpsize, wavout);
+
+        fclose(wavout);
+
+        if(WAV_DMABUF != NULL) free(WAV_DMABUF);
+    }
+}
+
+
+static void WAV_Update(void)
+{
+    VC_WriteBytes(WAV_DMABUF, WAVBUFFERSIZE);
+    fwrite(WAV_DMABUF, 1, WAVBUFFERSIZE, wavout);
+    dumpsize += WAVBUFFERSIZE;
+}
+
+
+static BOOL WAV_Reset(void)
+{
+    return 0;
+}
+
+
+MDRIVER drv_wav =
+{   NULL,
+    "music.wav file",
+    "WAV [music.wav] file output driver v1.0",
+    0,255,
+    WAV_IsThere,
+    VC_SampleLoad,
+    VC_SampleUnload,
+    VC_SampleSpace,
+    VC_SampleLength,
+    WAV_Init,
+    WAV_Exit,
+    WAV_Reset,
+    VC_SetNumVoices,
+    VC_PlayStart,
+    VC_PlayStop,
+    WAV_Update,
+    VC_VoiceSetVolume,
+    VC_VoiceSetFrequency,
+    VC_VoiceSetPanning,
+    VC_VoicePlay,
+    VC_VoiceStop,
+    VC_VoiceStopped,
+    VC_VoiceReleaseSustain,
+    VC_VoiceGetPosition,
+    NULL
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_669.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,255 @@
+/*
+
+ Name: LOAD_669.C
+
+ Description:
+ Tran's 669 module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+
+// Raw 669 header struct:
+
+typedef struct S69HEADER
+{   UBYTE  marker[2];
+    CHAR   message[108];
+    UBYTE  nos;
+    UBYTE  nop;
+    UBYTE  looporder;
+    UBYTE  orders[0x80];
+    UBYTE  tempos[0x80];
+    UBYTE  breaks[0x80];
+} S69HEADER;
+
+
+// Raw 669 sampleinfo struct:
+
+typedef struct S69SAMPLE
+{   CHAR   filename[13];
+    SLONG  length;
+    SLONG  loopbeg;
+    SLONG  loopend;
+} S69SAMPLE;
+
+
+// Raw 669 Note struct
+
+typedef struct S69NOTE
+{   UBYTE a,b,c;
+} S69NOTE;
+
+
+static S69NOTE *s69pat = NULL;
+static S69HEADER *mh   = NULL;
+
+static CHAR *S69_Version[] =
+{   "669",
+    "Extended 669"
+};
+
+
+BOOL S69_Test(void)
+{
+    UBYTE id[2];
+
+    if(!_mm_read_UBYTES(id,2,modfp)) return 0;
+    if(!memcmp(id,"if",2) || !memcmp(id,"JN",2))
+    {   _mm_fseek(modfp,108,SEEK_CUR);
+        if(_mm_read_UBYTE(modfp) > 64) return 0;
+        if(_mm_read_UBYTE(modfp) > 128) return 0;
+        if(_mm_read_UBYTE(modfp) > 120) return 0;
+        return 1;
+    }
+    return 0;
+}
+
+
+BOOL S69_Init(void)
+{
+    if(!(s69pat=(S69NOTE *)_mm_malloc(64*8*sizeof(S69NOTE)))) return 0;
+    if(!(mh=(S69HEADER *)_mm_calloc(1,sizeof(S69HEADER)))) return 0;
+    return 1;
+}
+
+
+void S69_Cleanup(void)
+{
+    if(s69pat!=NULL) free(s69pat);
+    if(mh!=NULL) free(mh);
+
+    mh     = NULL;
+    s69pat = NULL;
+}
+
+
+BOOL S69_LoadPatterns(void)
+{
+    int     t,s,q,tracks=0,t2,t3;
+    UBYTE   note,inst,vol,a,b,c, lo;
+    S69NOTE *cur;
+
+    if(!AllocPatterns()) return 0;
+    if(!AllocTracks())   return 0;
+
+    for(t=0; t<of.numpat; t++)
+    {   of.pattrows[t] = mh->breaks[t]+1;
+
+        // Load the pattern into the temp buffer
+        // and convert it into the 3-byte format
+
+        cur = s69pat;
+        for(t2=64; t2; t2--)
+        {   for(t3=8; t3; t3--, cur++)
+            {   cur->a = _mm_read_UBYTE(modfp);
+                cur->b = _mm_read_UBYTE(modfp);
+                cur->c = _mm_read_UBYTE(modfp);
+            }
+        }
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 0;
+        }
+
+        for(s=0; s<8; s++)
+        {   UniReset();
+            UniPTEffect(0xf,75);   // was 78
+            UniPTEffect(0xf,3);
+
+            for(q=0; q<64; q++)
+            {   a = s69pat[(q*8)+s].a;
+                b = s69pat[(q*8)+s].b;
+                c = s69pat[(q*8)+s].c;
+
+                note = a >> 2;
+                inst = ((a & 0x3) << 4) | ((b & 0xf0) >> 4);
+                vol  = b & 0xf;
+
+                if(note < 0x3e)
+                {   UniInstrument(inst);
+                    UniNote(note+24);
+                }
+
+                if(note < 0x3f) UniPTEffect(0xc,vol<<2);
+
+                lo = c & 0xf;
+                switch(c >> 4)
+                {   case 0:
+                        UniPTEffect(0x1,lo);
+                    break;
+
+                    case 1:
+                        UniPTEffect(0x2,lo);
+                    break;
+                   
+                    case 2:
+                        UniPTEffect(0x3,lo);
+                    break;
+
+                    case 4:
+                        UniPTEffect(0x4,lo);
+                    break;
+                }
+                UniNewline();
+            }
+            if(!(of.tracks[tracks++]=UniDup())) return 0;
+        }
+    }
+    return 1;
+}
+
+
+BOOL S69_Load(void)
+{
+    int         t;
+    S69SAMPLE   s;
+    SAMPLE     *q;
+
+    // try to read module header       
+
+    _mm_read_UBYTES(mh->marker,2,modfp);
+    _mm_read_UBYTES((UBYTE *)mh->message,108,modfp);
+    mh->nos = _mm_read_UBYTE(modfp);
+    mh->nop = _mm_read_UBYTE(modfp);
+    mh->looporder = _mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(mh->orders,0x80,modfp);
+    _mm_read_UBYTES(mh->tempos,0x80,modfp);
+    _mm_read_UBYTES(mh->breaks,0x80,modfp);
+
+    // set module variables
+
+    of.initspeed = 6;
+    of.inittempo = 125;
+    of.songname  = DupStr(mh->message,108);
+    of.modtype   = strdup(S69_Version[memcmp(mh->marker,"JN",2)==0]);
+    of.numchn    = 8;
+    of.numpat    = mh->nop;
+    of.numins    = of.numsmp = mh->nos;
+    of.numtrk    = of.numchn*of.numpat;
+    of.flags     = UF_XMPERIODS; // | UF_LINEAR;
+
+    if(!AllocPositions(0x80)) return 0;
+    for(t=0; t<0x80; t++)
+    {   if(mh->orders[t]==0xff) break;
+        of.positions[t] = mh->orders[t];
+    }
+
+    of.numpos = t;
+
+    if(!AllocSamples()) return 0;
+    q = of.samples;
+
+    for(t=0; t<of.numins; t++)
+    {   // try to read sample info
+
+        _mm_read_UBYTES((UBYTE *)s.filename,13,modfp);
+        s.length   = _mm_read_I_SLONG(modfp);
+        s.loopbeg  = _mm_read_I_SLONG(modfp);
+        s.loopend  = _mm_read_I_SLONG(modfp);
+
+        if((s.length < 0) || (s.loopbeg < -1) || (s.loopend < -1))
+        {   _mm_errno = MMERR_LOADING_HEADER;
+            return 0;
+        }
+
+        q->samplename = DupStr(s.filename,13);
+
+        q->seekpos   = 0;
+        q->speed     = 0;
+        q->length    = s.length;
+        q->loopstart = s.loopbeg;
+        q->loopend   = (s.loopend<s.length) ? s.loopend : s.length;
+        q->flags     = (s.loopbeg<s.loopend) ? SF_LOOP : 0;
+        q->volume    = 64;
+
+        q++;
+    }
+
+    if(!S69_LoadPatterns()) return 0;
+
+    return 1;
+}
+
+
+MLOADER load_669 =
+{   NULL,
+    "669",
+    "Portable 669 loader v0.1",
+    S69_Init,
+    S69_Test,
+    S69_Load,
+    S69_Cleanup,
+    NULL
+};
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_dsm.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,314 @@
+/*
+
+ Name: LOAD_DSM.C
+
+ Description:
+ DSIK Internal Format (DSM) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+
+typedef struct DSMNOTE
+{   UBYTE note,ins,vol,cmd,inf;
+} DSMNOTE;
+
+
+typedef struct DSMINST
+{   CHAR  filename[13];
+    UWORD flags;
+    UBYTE volume;
+    ULONG length;
+    ULONG loopstart;
+    ULONG loopend;
+    ULONG reserved1;
+    UWORD c2spd;
+    UWORD reserved2;
+    CHAR  samplename[28];
+} DSMINST;
+
+
+typedef struct DSMSONG
+{  CHAR  songname[28];
+   UWORD reserved1;
+   UWORD flags;
+   ULONG reserved2;
+   UWORD numord;
+   UWORD numsmp;
+   UWORD numpat;
+   UWORD numtrk;
+   UBYTE globalvol;
+   UBYTE mastervol;
+   UBYTE speed;
+   UBYTE bpm;
+   UBYTE panpos[16];
+   UBYTE orders[128];
+} DSMSONG;
+
+
+
+static CHAR  *SONGID = "SONG";
+static CHAR  *INSTID = "INST";
+static CHAR  *PATTID = "PATT";
+
+
+static UBYTE blockid[4];
+static ULONG blockln;
+static ULONG blocklp;
+static DSMSONG *mh = NULL;
+static DSMNOTE *dsmbuf = NULL;
+
+static CHAR  DSM_Version[] = "DSIK DSM-format";
+
+
+BOOL DSM_Test(void)
+{
+    UBYTE id[12];
+
+    if(_mm_read_UBYTES((UBYTE *)id,12,modfp)) return 0;
+    if(!memcmp(id,"RIFF",4) && !memcmp(&id[8],"DSMF",4)) return 1;
+    
+    return 0;
+}
+
+
+BOOL DSM_Init(void)
+{
+    if(!(dsmbuf=(DSMNOTE *)_mm_malloc(16*64*sizeof(DSMNOTE)))) return 0;
+    if(!(mh=(DSMSONG *)_mm_calloc(1,sizeof(DSMSONG)))) return 0;
+    return 1;
+}
+
+
+void DSM_Cleanup(void)
+{
+   if(dsmbuf!=NULL) free(dsmbuf);
+   if(mh!=NULL) free(mh);
+
+   dsmbuf = NULL;
+   mh     = NULL;
+}
+
+
+BOOL GetBlockHeader(void)
+{
+   // make sure we're at the right position for reading the
+   // next riff block, no matter how many bytes read
+
+   _mm_fseek(modfp, blocklp+blockln, SEEK_SET);
+   
+   while(1)
+   {   _mm_read_UBYTES(blockid,4,modfp);
+       blockln = _mm_read_I_ULONG(modfp);
+       if(feof(modfp))
+       {   _mm_errno = MMERR_LOADING_HEADER;
+           return 0;
+       }
+       
+       if(memcmp(blockid,SONGID,4) && memcmp(blockid,INSTID,4) && memcmp(blockid,PATTID,4))
+       {   //printf("Skipping unknown block type %4.4s\n",&blockid);
+           _mm_fseek(modfp, blockln, SEEK_CUR);
+       } else break;
+   }
+
+   blocklp = _mm_ftell(modfp);
+   return 1;
+}
+
+
+BOOL DSM_ReadPattern(void)
+{
+    int row=0,flag;
+    DSMNOTE *n;
+
+    // clear pattern data
+    memset(dsmbuf,255,16*64*sizeof(DSMNOTE));
+    _mm_read_UBYTE(modfp);
+    _mm_read_UBYTE(modfp);
+
+    while(row<64)
+    {   flag = _mm_read_UBYTE(modfp);
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 0;
+        }
+
+        if(flag)
+        {   n = &dsmbuf[((flag&0xf)*64)+row];
+            if(flag&0x80) n->note = _mm_read_UBYTE(modfp);
+            if(flag&0x40) n->ins  = _mm_read_UBYTE(modfp);
+            if(flag&0x20) n->vol  = _mm_read_UBYTE(modfp);
+            if(flag&0x10)
+            {   n->cmd = _mm_read_UBYTE(modfp);
+                n->inf = _mm_read_UBYTE(modfp);
+            }
+        } else row++;
+    }
+    return 1;
+}
+
+
+UBYTE *DSM_ConvertTrack(DSMNOTE *tr)
+{
+    int t;
+    UBYTE note,ins,vol,cmd,inf;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {   note = tr[t].note;
+        ins  = tr[t].ins;
+        vol  = tr[t].vol;
+        cmd  = tr[t].cmd;
+        inf  = tr[t].inf;
+
+        if(ins!=0 && ins!=255) UniInstrument(ins-1);
+        if(note!=255) UniNote(note-1);                // <- normal note
+        if(vol<65) UniPTEffect(0xc,vol);
+
+        if(cmd!=255)
+        {   if(cmd==0x8)
+            {   if(inf<=0x80)
+                {   inf = (inf<0x80) ? inf<<1 : 255;
+                    UniPTEffect(cmd,inf);
+                }
+            } else if(cmd==0xb)
+            {   if(inf<=0x7f) UniPTEffect(cmd,inf);
+            } else
+            {   // Convert pattern jump from Dec to Hex
+                if(cmd == 0xd)
+                    inf = (((inf&0xf0)>>4)*10)+(inf&0xf);
+                UniPTEffect(cmd,inf);
+            }
+        }
+        UniNewline();
+     }
+     return UniDup();
+}
+
+
+BOOL DSM_Load(void)
+{
+       int     t;
+       DSMINST s;
+       SAMPLE *q;
+       int    cursmp = 0, curpat = 0, track = 0;
+
+       blocklp = 0;
+       blockln = 12;
+
+       if(!GetBlockHeader()) return 0;
+       if(memcmp(blockid,SONGID,4))
+       {   _mm_errno = MMERR_LOADING_HEADER;
+           return 0;
+       }
+
+       _mm_read_UBYTES(mh->songname,28,modfp);
+       mh->reserved1   = _mm_read_I_UWORD(modfp);
+       mh->flags       = _mm_read_I_UWORD(modfp);
+       mh->reserved2   = _mm_read_I_ULONG(modfp);
+       mh->numord      = _mm_read_I_UWORD(modfp);
+       mh->numsmp      = _mm_read_I_UWORD(modfp);
+       mh->numpat      = _mm_read_I_UWORD(modfp);
+       mh->numtrk      = _mm_read_I_UWORD(modfp);
+       mh->globalvol   = _mm_read_UBYTE(modfp);
+       mh->mastervol   = _mm_read_UBYTE(modfp);
+       mh->speed       = _mm_read_UBYTE(modfp);
+       mh->bpm         = _mm_read_UBYTE(modfp);
+       _mm_read_UBYTES(mh->panpos,16,modfp);
+       _mm_read_UBYTES(mh->orders,128,modfp);
+
+       // set module variables
+       of.initspeed = mh->speed;
+       of.inittempo = mh->bpm;
+       of.modtype   = strdup(DSM_Version);
+       of.numchn    = mh->numtrk;
+       of.numpat    = mh->numpat;
+       of.numtrk    = of.numchn*of.numpat;
+       of.songname  = DupStr(mh->songname,28);    // make a cstr of songname
+
+       for(t=0; t<16; t++)
+          of.panning[t] = mh->panpos[t]<0x80 ? (mh->panpos[t]<<1) : 255;
+
+       if(!AllocPositions(mh->numord)) return 0;
+       of.numpos = 0;
+       for(t=0; t<mh->numord; t++)
+       {   of.positions[of.numpos] = mh->orders[t];
+           if(mh->orders[t]<254) of.numpos++;
+       }
+
+       of.numins = of.numsmp = mh->numsmp;
+
+       if(!AllocSamples()) return 0;
+       if(!AllocTracks()) return 0;
+       if(!AllocPatterns()) return 0;
+
+       while(cursmp<of.numins || curpat<of.numpat)
+       {   if(!GetBlockHeader()) return 0;
+           if(!memcmp(blockid,INSTID,4) && cursmp<of.numins)
+           {   q = &of.samples[cursmp];
+               
+               // try to read sample info
+               _mm_read_UBYTES(s.filename,13,modfp);
+               s.flags     = _mm_read_I_UWORD(modfp);
+               s.volume    = _mm_read_UBYTE(modfp);
+               s.length    = _mm_read_I_ULONG(modfp);
+               s.loopstart = _mm_read_I_ULONG(modfp);
+               s.loopend   = _mm_read_I_ULONG(modfp);
+               s.reserved1 = _mm_read_I_ULONG(modfp);
+               s.c2spd     = _mm_read_I_UWORD(modfp);
+               s.reserved2 = _mm_read_I_UWORD(modfp);
+               _mm_read_UBYTES(s.samplename,28,modfp);
+
+               q->samplename= DupStr(s.samplename,28);
+               q->seekpos   = _mm_ftell(modfp);
+               q->speed     = s.c2spd;
+               q->length    = s.length;
+               q->loopstart = s.loopstart;
+               q->loopend   = s.loopend;
+               q->volume    = s.volume;
+
+               if(s.flags&1) q->flags|=SF_LOOP;
+               if(s.flags&2) q->flags|=SF_SIGNED;
+               cursmp++;
+           } else if(!memcmp(blockid,PATTID,4) && curpat<of.numpat)
+           {   DSM_ReadPattern();
+               for(t=0; t<of.numchn; t++)
+                  if(!(of.tracks[track++] = DSM_ConvertTrack(&dsmbuf[t*64]))) return 0;
+               curpat++;
+           }
+       }
+
+       return 1;
+}
+
+
+CHAR *DSM_LoadTitle(void)
+{
+   CHAR s[28];
+
+   _mm_fseek(modfp,12,SEEK_SET);
+   if(!fread(s,28,1,modfp)) return NULL;
+   
+   return(DupStr(s,28));
+}
+
+
+MLOADER load_dsm =
+{   NULL,
+    "DSM",
+    "Portable DSM loader v0.1",
+    DSM_Init,
+    DSM_Test,
+    DSM_Load,
+    DSM_Cleanup,
+    DSM_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_far.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,288 @@
+/*
+
+ Name: LOAD_FAR.C
+
+ Description:
+ Farandole (FAR) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+
+typedef struct FARSAMPLE
+{   CHAR  samplename[32];
+    ULONG length;
+    UBYTE finetune;
+    UBYTE volume;
+    ULONG reppos;
+    ULONG repend;
+    UBYTE type;
+    UBYTE loop;
+} FARSAMPLE;
+
+
+
+typedef struct FARHEADER1
+{   UBYTE id[4];                        // file magic
+    CHAR  songname[40];                 // songname
+    CHAR  blah[3];                      // 13,10,26
+    UWORD headerlen;                    // remaining length of header in bytes
+    UBYTE version;
+    UBYTE onoff[16];
+    UBYTE edit1[9];
+    UBYTE speed;
+    UBYTE panning[16];
+    UBYTE edit2[4];
+    UWORD stlen;
+} FARHEADER1;
+
+
+typedef struct FARHEADER2
+{   UBYTE orders[256];
+    UBYTE numpat;
+    UBYTE snglen;
+    UBYTE loopto;
+    UWORD patsiz[256];
+} FARHEADER2;
+
+
+typedef struct FARNOTE
+{   UBYTE note,ins,vol,eff;
+} FARNOTE;
+
+
+
+static CHAR FAR_Version[] = "Farandole";
+static FARHEADER1 *mh1 = NULL;
+static FARHEADER2 *mh2 = NULL;
+static FARNOTE *pat = NULL;
+
+
+BOOL FAR_Test(void)
+{
+    UBYTE id[4];
+
+    if(!_mm_read_UBYTES(id,4,modfp)) return 0;
+    return(!memcmp(id,"FAR=",4));
+}
+
+
+BOOL FAR_Init(void)
+{
+    if(!(mh1 = (FARHEADER1 *)_mm_malloc(sizeof(FARHEADER1)))) return 0;
+    if(!(mh2 = (FARHEADER2 *)_mm_malloc(sizeof(FARHEADER2)))) return 0;
+    if(!(pat = (FARNOTE *)_mm_malloc(16*256*sizeof(FARNOTE)))) return 0;
+
+    return 1;
+}
+
+
+void FAR_Cleanup(void)
+{
+    if(mh1!=NULL) free(mh1);
+    if(mh2!=NULL) free(mh2);
+    if(pat!=NULL) free(pat);
+
+    mh1 = NULL;
+    mh2 = NULL;
+    pat = NULL;
+}
+
+
+UBYTE *FAR_ConvertTrack(FARNOTE *n,int rows)
+{
+       int t;
+
+       UniReset();
+
+       for(t=0; t<rows; t++)
+       {   if(n->note)
+           {   UniInstrument(n->ins);
+               UniNote(n->note+23+12);
+           }
+
+           if(n->vol&0xf) UniPTEffect(0xc,(n->vol&0xf)<<2);
+           switch(n->eff>>4)
+           {   case 0xf:
+                  UniPTEffect(0xf,n->eff&0xf);
+               break;
+
+               // others not yet implemented
+           }
+
+           UniNewline();
+           n+=16;
+       }
+       return UniDup();
+}
+
+
+BOOL FAR_Load(void)
+{
+       int t,u,tracks=0;
+       SAMPLE     *q;
+       FARSAMPLE  s;
+       FARNOTE    *crow;
+       UBYTE      smap[8];
+
+       // try to read module header (first part)
+       _mm_read_UBYTES(mh1->id,4,modfp);
+       _mm_read_SBYTES(mh1->songname,40,modfp);
+       _mm_read_SBYTES(mh1->blah,3,modfp);
+       mh1->headerlen = _mm_read_I_UWORD (modfp);
+       mh1->version   = _mm_read_UBYTE (modfp);
+       _mm_read_UBYTES(mh1->onoff,16,modfp);
+       _mm_read_UBYTES(mh1->edit1,9,modfp);
+       mh1->speed     = _mm_read_UBYTE(modfp);
+       _mm_read_UBYTES(mh1->panning,16,modfp);
+       _mm_read_UBYTES(mh1->edit2,4,modfp);
+       mh1->stlen     = _mm_read_I_UWORD (modfp);
+
+
+       // init modfile data
+
+       of.modtype   = strdup(FAR_Version);
+       of.songname  = DupStr(mh1->songname,40);
+       of.numchn    = 16;
+       of.initspeed = mh1->speed;
+       of.inittempo = 99;
+
+       for(t=0; t<16; t++) of.panning[t] = mh1->panning[t]<<4;
+
+       // read songtext into comment field
+       if(!ReadComment(mh1->stlen)) return 0;
+
+       // try to read module header (second part)
+       _mm_read_UBYTES(mh2->orders,256,modfp);
+       mh2->numpat        = _mm_read_UBYTE(modfp);
+       mh2->snglen        = _mm_read_UBYTE(modfp);
+       mh2->loopto        = _mm_read_UBYTE(modfp);
+       _mm_read_I_UWORDS(mh2->patsiz,256,modfp);
+
+//      of.numpat=mh2->numpat;
+       of.numpos = mh2->snglen;
+       if(!AllocPositions(of.numpos)) return 0;
+       for(t=0; t<of.numpos; t++)
+       {   if(mh2->orders[t]==0xff) break;
+           of.positions[t] = mh2->orders[t];
+       }
+
+       // count number of patterns stored in file
+       of.numpat = 0;
+       for(t=0; t<256; t++)
+          if(mh2->patsiz[t]) if((t+1)>of.numpat) of.numpat=t+1;
+
+       of.numtrk = of.numpat*of.numchn;
+
+       // seek across eventual new data
+       _mm_fseek(modfp,mh1->headerlen-(869+mh1->stlen),SEEK_CUR);
+
+       // alloc track and pattern structures
+       if(!AllocTracks()) return 0;
+       if(!AllocPatterns()) return 0;
+
+       for(t=0; t<of.numpat; t++)
+       {   UBYTE rows=0,tempo;
+
+           memset(pat,0,16*256*sizeof(FARNOTE));
+           if(mh2->patsiz[t])
+           {   rows  = _mm_read_UBYTE(modfp);
+               tempo = _mm_read_UBYTE(modfp);
+
+               crow = pat;
+               for(u=mh2->patsiz[t]-2; u; u--, crow++)
+               {   crow->note = _mm_read_UBYTE(modfp);
+                   crow->ins  = _mm_read_UBYTE(modfp);
+                   crow->vol  = _mm_read_UBYTE(modfp);
+                   crow->eff  = _mm_read_UBYTE(modfp);
+               }
+               
+               if(feof(modfp))
+               {   _mm_errno = MMERR_LOADING_PATTERN;
+                   return 0;
+               }
+
+               of.pattrows[t] = rows+2;
+               crow = pat;
+               for(u=16; u; u--)
+                  if(!(of.tracks[tracks++] = FAR_ConvertTrack(crow,rows+2))) return 0;
+           }
+       }
+
+       // read sample map
+       if(!_mm_read_UBYTES(smap,8,modfp))
+       {   _mm_errno = MMERR_LOADING_HEADER;
+           return 0;
+       }
+
+       // count number of samples used
+       of.numins = 0;
+       for(t=0; t<64; t++)
+          if(smap[t>>3] & (1 << (t&7)))  of.numins++;
+       of.numsmp = of.numins;             
+
+       // alloc sample structs
+       if(!AllocSamples()) return 0;
+       q = of.samples;
+
+       for(t=0; t<64; t++)
+       {  if(smap[t>>3] & (1 << (t&7)))
+           {  _mm_read_SBYTES(s.samplename,32,modfp);
+              s.length   = _mm_read_I_ULONG(modfp);
+              s.finetune = _mm_read_UBYTE(modfp);
+              s.volume   = _mm_read_UBYTE(modfp);
+              s.reppos   = _mm_read_I_ULONG(modfp);
+              s.repend   = _mm_read_I_ULONG(modfp);
+              s.type     = _mm_read_UBYTE(modfp);
+              s.loop     = _mm_read_UBYTE(modfp);
+
+              q->samplename = DupStr(s.samplename,32);
+              q->length     = s.length;
+              q->loopstart  = s.reppos;
+              q->loopend    = s.repend;
+              q->volume     = 64;
+              q->speed      = 8363;
+
+              q->flags=SF_SIGNED;
+              if(s.type&1) q->flags|=SF_16BITS;
+              if(s.loop) q->flags|=SF_LOOP;
+
+              q->seekpos = _mm_ftell(modfp);
+              _mm_fseek(modfp,q->length,SEEK_CUR);
+           }
+           q++;
+       }
+       return 1;
+}
+
+
+CHAR *FAR_LoadTitle(void)
+{
+   CHAR s[40];
+
+   _mm_fseek(modfp,4,SEEK_SET);
+   if(!fread(s,40,1,modfp)) return NULL;
+   
+   return(DupStr(s,40));
+}
+
+
+MLOADER load_far =
+{   NULL,
+    "FAR",
+    "Portable FAR loader v0.1",
+    FAR_Init,
+    FAR_Test,
+    FAR_Load,
+    FAR_Cleanup,
+    FAR_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_it.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,857 @@
+/*
+
+ Name: LOAD_IT.C
+
+ Description:
+ ImpulseTracker (IT) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ Copyright 1997 by Jake Stine and Divine Entertainment
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/**************************************************************************
+**************************************************************************/
+
+typedef struct ITNOTE
+{   UBYTE note,ins,volpan,cmd,inf;
+} ITNOTE;
+
+UBYTE *IT_ConvertTrack(ITNOTE *tr,UWORD numrows);
+
+
+// Raw IT header struct:
+
+typedef struct ITHEADER
+{   CHAR  songname[26];
+    UBYTE blank01[2];
+    UWORD ordnum;
+    UWORD insnum;
+    UWORD smpnum;
+    UWORD patnum;
+    UWORD cwt;          // Created with tracker (y.xx = 0x0yxx)
+    UWORD cmwt;         // Compatable with tracker ver > than val.
+    UWORD flags;
+    UWORD special;      // bit 0 set = song message attached
+    UBYTE globvol;
+    UBYTE mixvol;       // mixing volume [ignored]
+    UBYTE initspeed;
+    UBYTE inittempo;
+    UBYTE pansep;       // panning separation between channels
+    UBYTE zerobyte;       
+    UWORD msglength;
+    ULONG msgoffset;
+    UBYTE blank02[4];
+ 
+    UBYTE pantable[64];
+    UBYTE voltable[64];
+} ITHEADER;
+
+
+// Raw IT sampleinfo struct:
+
+typedef struct ITSAMPLE
+{   CHAR  filename[12];
+    UBYTE zerobyte;
+    UBYTE globvol;
+    UBYTE flag;
+    UBYTE volume;
+    UBYTE panning;
+    CHAR  sampname[28];
+    UWORD convert;        // sample conversion flag
+    ULONG length;
+    ULONG loopbeg;
+    ULONG loopend;
+    ULONG c5spd;
+    ULONG susbegin;
+    ULONG susend;
+    ULONG sampoffset;
+    UBYTE vibspeed;
+    UBYTE vibdepth;
+    UBYTE vibrate;
+    UBYTE vibwave;    // 0 = sine; 1 = rampdown; 2 = square; 3 = random (speed ignored)
+
+    UBYTE noteindex;  // for converting c5spd to finetune
+} ITSAMPLE;
+
+
+typedef struct ITINSTHEADER
+{   ULONG size;             // (dword) Instrument size
+    CHAR  filename[12];     // (char) Instrument filename
+    UBYTE zerobyte;         // (byte) Instrument type (always 0)
+    UBYTE volflg;
+    UBYTE volpts;   
+    UBYTE volbeg;           // (byte) Volume loop start (node)
+    UBYTE volend;           // (byte) Volume loop end (node)
+    UBYTE volsusbeg;        // (byte) Volume sustain begin (node)
+    UBYTE volsusend;        // (byte) Volume Sustain end (node)
+    UBYTE panflg;
+    UBYTE panpts;  
+    UBYTE panbeg;           // (byte) channel loop start (node)
+    UBYTE panend;           // (byte) channel loop end (node)
+    UBYTE pansusbeg;        // (byte) cahnnel sustain begin (node)
+    UBYTE pansusend;        // (byte) channel Sustain end (node)
+    UBYTE pitflg;
+    UBYTE pitpts;   
+    UBYTE pitbeg;           // (byte) pitch loop start (node)
+    UBYTE pitend;           // (byte) pitch loop end (node)
+    UBYTE pitsusbeg;        // (byte) pitch sustain begin (node)
+    UBYTE pitsusend;        // (byte) pitch Sustain end (node)
+    UWORD blank;
+    UBYTE globvol;
+    UBYTE chanpan;
+    UWORD fadeout;          // Envelope end / NNA volume fadeout
+    UBYTE dnc;              // Duplicate note check
+    UBYTE dca;              // Duplicate check action
+    UBYTE dct;              // Duplicate check type
+    UBYTE nna;              // New Note Action [0,1,2,3]
+    UWORD trkvers;          // tracker version used to save [in files only]
+    UBYTE ppsep;            // Pitch-pan Separation
+    UBYTE ppcenter;         // Pitch-pan Center
+    UBYTE rvolvar;          // random volume varations
+    UBYTE rpanvar;          // random panning varations
+    UWORD numsmp;           // Number of samples in instrument [in files only]
+    CHAR  name[26];         // Instrument name
+    UBYTE blank01[6];
+    UWORD samptable[120];   // sample for each note [note / samp pairs]
+
+    UBYTE volenv[200];      // volume envelope (IT 1.x stuff)
+    UBYTE oldvoltick[25];   // volume tick position (IT 1.x stuff)
+    UBYTE volnode[25];      // aplitude of volume nodes
+    UWORD voltick[25];      // tick value of volume nodes
+    SBYTE pannode[25];      // panenv - node points
+    UWORD pantick[25];      // tick value of panning nodes
+    SBYTE pitnode[25];      // pitchenv - node points
+    UWORD pittick[25];      // tick value of pitch nodes
+} ITINSTHEADER;                       
+
+
+/**************************************************************************
+**************************************************************************/
+
+extern SBYTE    remap[64];        // for removing empty channels
+extern UBYTE    *poslookup;       // S3M/IT fix - removing blank patterns needs a
+                                  // lookup table to fix position-jump commands
+static ULONG    *paraptr = NULL;  // parapointer array (see IT docs)
+static ITHEADER *mh = NULL;
+static ITNOTE   *itpat = NULL;    // allocate to space for one full pattern
+static UBYTE    *mask = NULL;     // arrays allocated to 64 elements and used for
+static ITNOTE   *last = NULL;     // uncompressing IT's pattern information
+static int       numtrk = 0;
+static int       old_effect;      // if set, use S3M old-effects stuffs
+static int       *noteindex;
+   
+CHAR IT_Version[] = "ImpulseTracker x.xx";
+
+
+BOOL IT_Test(void)
+{
+    UBYTE id[4];
+    
+    if(!_mm_read_UBYTES(id,4,modfp)) return 0;
+    if(!memcmp(id,"IMPM",4)) return 1;
+    return 0;
+}
+
+BOOL IT_Init(void)
+{
+    if((mh=(ITHEADER *)_mm_calloc(1,sizeof(ITHEADER)))==NULL) return 0;
+    if((poslookup=(UBYTE *)_mm_malloc(256*sizeof(UBYTE)))==NULL) return 0;
+    if((itpat=(ITNOTE *)_mm_malloc(200*64*sizeof(ITNOTE)))==NULL) return 0;
+    if((mask=(UBYTE *)_mm_malloc(64*sizeof(UBYTE)))==NULL) return 0;
+    if((last=(ITNOTE *)_mm_malloc(64*sizeof(ITNOTE)))==NULL) return 0;
+    
+    return 1;
+}
+
+void IT_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    if(poslookup!=NULL) free(poslookup);
+    if(itpat!=NULL) free(itpat);
+    if(mask!=NULL) free(mask);
+    if(last!=NULL) free(last);
+    if(paraptr!=NULL) free(paraptr);
+    if(noteindex!=NULL) free(noteindex);
+
+    mh        = NULL;
+    poslookup = NULL;
+    itpat     = NULL;
+    mask      = NULL;
+    last      = NULL;
+    paraptr   = NULL;
+    noteindex = NULL;
+}
+
+
+BOOL IT_GetNumChannels(UWORD patrows)
+
+// Because so many IT files have 64 channels as the set number used, but really
+// only use far less (usually 8 to 12 still), I had to make this function,
+// which determines the number of channels that are actually USED by a pattern.
+//
+// For every channel that's used, it sets the appropriate array entry of the
+// global varialbe 'isused'
+//
+// NOTE: You must first seek to the file location of the pattern before calling
+//       this procedure.
+// Returns 1 on error
+{
+    int  row=0,flag,ch;
+
+    do
+    {   flag = _mm_read_UBYTE(modfp);
+        if(flag == EOF)
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 1;
+        }
+
+        if(flag == 0)
+        {   row++;
+        } else
+        {   ch = (flag-1) & 63;
+            remap[ch] = 0;
+            if(flag & 128) mask[ch] = _mm_read_UBYTE(modfp);
+            if(mask[ch] &   1) _mm_read_UBYTE(modfp);
+            if(mask[ch] &   2) _mm_read_UBYTE(modfp);
+            if(mask[ch] &   4) _mm_read_UBYTE(modfp);
+            if(mask[ch] &   8) { _mm_read_UBYTE(modfp); _mm_read_UBYTE(modfp); }
+        }
+    } while(row < patrows);
+
+    return 0;
+}
+
+
+BOOL IT_ReadPattern(UWORD patrows)
+{
+   int    blah;
+   int    row=0,flag,ch;
+   ITNOTE *itt = itpat, dummy,*n,*l;
+
+   memset(itt,255,patrows*of.numchn*sizeof(ITNOTE));
+
+   do 
+   {   flag = _mm_read_UBYTE(modfp);
+       if(feof(modfp))
+       {   _mm_errno = MMERR_LOADING_PATTERN;
+           return 0;
+       }
+       if(flag == 0)
+       {   itt = &itt[of.numchn];
+           row++;
+       } else
+       {   ch = remap[(flag-1) & 63];
+           if(ch != -1)
+           {   n = &itt[ch];
+               l = &last[ch];
+           } else
+           {   n = l = &dummy;  }
+
+           if(flag & 128) mask[ch] = _mm_read_UBYTE(modfp);
+           if(mask[ch] &   1) if((l->note = n->note   = _mm_read_UBYTE(modfp)) == 255)
+                              {   l->note = n->note   = 253; }
+           if(mask[ch] &   2) l->ins      = n->ins    = _mm_read_UBYTE(modfp);
+           if(mask[ch] &   4) l->volpan   = n->volpan = _mm_read_UBYTE(modfp);
+           if(mask[ch] &   8) { l->cmd    = n->cmd    = _mm_read_UBYTE(modfp);
+                                l->inf    = n->inf    = _mm_read_UBYTE(modfp); }
+           if(mask[ch] &  16) n->note     = l->note;
+           if(mask[ch] &  32) n->ins      = l->ins;
+           if(mask[ch] &  64) n->volpan   = l->volpan;
+           if(mask[ch] & 128) { n->cmd    = l->cmd;
+                                n->inf    = l->inf; }
+        }
+   } while(row < patrows);
+
+   for(blah=0; blah<of.numchn; blah++)
+       of.tracks[numtrk++] = IT_ConvertTrack(&itpat[blah],patrows);
+
+   return 1;
+}
+
+
+void S3MIT_ProcessCmd(UBYTE cmd, UBYTE inf, BOOL oldeffect);
+
+// table for porta-to-note command within volume/panning column
+static UBYTE portatable[] = "1, 4, 8, 16, 32, 64, 96, 128, 255";
+
+UBYTE *IT_ConvertTrack(ITNOTE *tr, UWORD numrows)
+{
+    int t;
+
+    UBYTE note,ins,volpan;
+
+    UniReset();
+
+    for(t=0; t<numrows; t++)
+    {   note    = tr[t*of.numchn].note;
+        ins     = tr[t*of.numchn].ins;
+        volpan  = tr[t*of.numchn].volpan;
+
+        if(note!=255)
+        {   if(note==253)
+                UniWrite(UNI_KEYOFF);
+            else if(note==254)
+                UniPTEffect(0xc,0);   // <- note cut command
+            else
+                UniNote(note);
+        }
+
+        if((ins != 0) && (ins < 100)) UniInstrument(ins-1);
+
+        // process volume / panning column
+        // volume / panning effects do NOT all share the same memory address
+        // yet.  That requires more work than I care to apply at the moment ;)
+
+        if(volpan<=64)
+        {   UniVolEffect(VOL_VOLUME,volpan);
+        } else if((volpan>=65) && (volpan<=74))        // fine volume slide up (65-74)
+        {   UniVolEffect(VOL_VOLSLIDE,0x0f + ((volpan-65)<<4));
+        } else if((volpan>=75) && (volpan<=84))        // fine volume slide down   (75-84)
+        {   UniVolEffect(VOL_VOLSLIDE,0xf0 + (volpan-75));
+        } else if((volpan>=85) && (volpan<=94))        // volume slide up (85-94)
+        {   UniVolEffect(VOL_VOLSLIDE,((volpan-85)<<4));
+        } else if((volpan>=95) && (volpan<=104))       // volume slide down   (95-104)
+        {   UniVolEffect(VOL_VOLSLIDE,(volpan-95));
+        } else if((volpan>=105) && (volpan<=114))      // pitch slide up (105-114)
+        {   UniVolEffect(VOL_PITCHSLIDEDN,((volpan-105)<<4));
+        } else if((volpan>=115) && (volpan<=124))      // pitch slide down (115-124)
+        {   UniVolEffect(VOL_PITCHSLIDEUP,(volpan-115));
+        } else if((volpan>=128) && (volpan<=192))
+        {   UniVolEffect(VOL_PANNING,((volpan-128) == 64) ? 255 : ((volpan-128) << 2));
+        } else if((volpan>=193) && (volpan<=202))      // portamento to note
+        {   UniVolEffect(VOL_PORTAMENTO,portatable[volpan-193]);
+        } else if((volpan>=203) && (volpan<=212))      // vibrato
+        {   UniVolEffect(VOL_VIBRATO,(volpan-203));
+        }
+
+        S3MIT_ProcessCmd(tr[t*of.numchn].cmd,tr[t*of.numchn].inf,old_effect);
+
+        UniNewline();
+    }
+    return UniDup();
+}
+                                             
+
+int cvt_c5spd_to_finetune(ULONG c5spd, int sampnum)
+{
+    int ctmp=0,tmp,note=1,finetune=0;
+
+    c5spd/=2;
+
+    do
+    {   tmp = getfrequency(of.flags,getlinearperiod(note,0));
+        if(tmp >= c5spd) break;
+        ctmp = tmp;
+        note++;
+    } while(1);
+
+    if(tmp != c5spd)
+    {   if((tmp-c5spd) < (c5spd-ctmp))
+            while(tmp>c5spd) tmp = getfrequency(of.flags,getlinearperiod(note,--finetune));
+        else
+        {   note--;
+            while(ctmp<c5spd) ctmp = getfrequency(of.flags,getlinearperiod(note,++finetune));
+        }
+    }
+
+    noteindex[sampnum] = note-48;
+    return finetune;
+}       
+
+
+BOOL IT_Load(void)
+{
+    int        t,u,lp;
+    INSTRUMENT *d;
+    SAMPLE     *q;
+
+    numtrk = 0;
+
+    // try to read module header
+
+    _mm_read_I_ULONG(modfp);   // kill the 4 byte header
+    _mm_read_string(mh->songname,26,modfp);
+    _mm_read_UBYTES(mh->blank01,2,modfp);
+    mh->ordnum      =_mm_read_I_UWORD(modfp);
+    mh->insnum      =_mm_read_I_UWORD(modfp);
+    mh->smpnum      =_mm_read_I_UWORD(modfp);
+    mh->patnum      =_mm_read_I_UWORD(modfp);
+    mh->cwt         =_mm_read_I_UWORD(modfp);
+    mh->cmwt        =_mm_read_I_UWORD(modfp);
+    mh->flags       =_mm_read_I_UWORD(modfp);
+    mh->special     =_mm_read_I_UWORD(modfp);
+
+    mh->globvol     =_mm_read_UBYTE(modfp);
+    mh->mixvol      =_mm_read_UBYTE(modfp);
+    mh->initspeed   =_mm_read_UBYTE(modfp);
+    mh->inittempo   =_mm_read_UBYTE(modfp);
+    mh->pansep      =_mm_read_UBYTE(modfp);
+    mh->zerobyte    =_mm_read_UBYTE(modfp);
+    mh->msglength   =_mm_read_I_UWORD(modfp);
+    mh->msgoffset   =_mm_read_I_ULONG(modfp);
+    _mm_read_UBYTES(mh->blank02,4,modfp);
+    _mm_read_UBYTES(mh->pantable,64,modfp);
+    _mm_read_UBYTES(mh->voltable,64,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+
+    of.modtype     = strdup(IT_Version);
+    of.modtype[15] = (mh->cwt >> 8) + 0x30;
+    of.modtype[17] = ((mh->cwt >> 4) & 0xf) + 0x30;
+    of.modtype[18] = ((mh->cwt) & 0xf) + 0x30;
+    of.songname    = DupStr(mh->songname,26);    // make a cstr of songname 
+    of.reppos      = 0;
+    of.numpat      = mh->patnum;
+    of.numins      = mh->insnum;
+    of.numsmp      = mh->smpnum;
+    of.initspeed   = mh->initspeed;
+    of.inittempo   = mh->inittempo;
+    of.initvolume  = mh->globvol;
+
+    old_effect = 0;
+    if(mh->flags & 8) { of.flags |= (UF_XMPERIODS | UF_LINEAR); old_effect |= 2; }
+    if((mh->cwt >= 0x106) && (mh->flags & 16))  old_effect |= 1;
+
+    // set panning positions
+    for(t=0; t<64; t++)
+    {   if(mh->pantable[t] < 64) of.panning[t] = mh->pantable[t] << 2;
+        else if(mh->pantable[t]==64) of.panning[t] = 255;
+        else if(mh->pantable[t]==100) of.panning[t] = PAN_SURROUND;
+    }
+
+    // set channel volumes
+    memcpy(of.chanvol,mh->voltable,64);
+
+    // read the order data
+    if(!AllocPositions(mh->ordnum)) return 0;
+
+    for(t=0; t<mh->ordnum; t++)
+        of.positions[t] = _mm_read_UBYTE(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+    
+    of.numpos = 0;
+    for(t=0; t<mh->ordnum; t++)
+    {   of.positions[of.numpos] = of.positions[t];
+        poslookup[t] = of.numpos;    // bug fix for FREAKY S3Ms / ITs
+        if(of.positions[t]<254) of.numpos++;
+    }
+
+    if((paraptr=(ULONG *)_mm_malloc((mh->insnum+mh->smpnum+of.numpat)*sizeof(ULONG))) == NULL) return 0;
+
+    // read the instrument, sample, and pattern parapointers
+    _mm_read_I_ULONGS(paraptr,mh->insnum+mh->smpnum+of.numpat,modfp);
+
+    // now is a good time to check if the header was too short :)
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // Check for and load song comment
+    if(mh->special & 1)
+    {   _mm_fseek(modfp,(long)(mh->msgoffset),SEEK_SET);
+        if(!ReadComment(mh->msglength)) return 0;
+    }
+
+    if(!(mh->flags & 4)) of.numins = of.numsmp;
+    if(!AllocSamples()) return 0;
+
+    if((noteindex=(int *)_mm_malloc(mh->smpnum*sizeof(int)))==NULL) return 0;
+        
+    q = of.samples;
+
+    // Load all samples (they're used either way)
+    for(t=0; t<mh->smpnum; t++)
+    {   ITSAMPLE s;
+
+        // seek to sample position
+        _mm_fseek(modfp,(long)(paraptr[mh->insnum+t] + 4),SEEK_SET);
+
+        // and load sample info
+        _mm_read_string(s.filename,12,modfp);
+        s.zerobyte    = _mm_read_UBYTE(modfp);
+        s.globvol     = _mm_read_UBYTE(modfp);
+        s.flag        = _mm_read_UBYTE(modfp);
+        s.volume      = _mm_read_UBYTE(modfp);
+        _mm_read_string(s.sampname,26,modfp);
+        s.convert     = _mm_read_UBYTE(modfp);
+        s.panning     = _mm_read_UBYTE(modfp);
+        s.length      = _mm_read_I_ULONG(modfp);
+        s.loopbeg     = _mm_read_I_ULONG(modfp);
+        s.loopend     = _mm_read_I_ULONG(modfp);
+        s.c5spd       = _mm_read_I_ULONG(modfp);
+        s.susbegin    = _mm_read_I_ULONG(modfp);
+        s.susend      = _mm_read_I_ULONG(modfp);
+        s.sampoffset  = _mm_read_I_ULONG(modfp);
+        s.vibspeed    = _mm_read_UBYTE(modfp);
+        s.vibdepth    = _mm_read_UBYTE(modfp);
+        s.vibrate     = _mm_read_UBYTE(modfp);
+        s.vibwave     = _mm_read_UBYTE(modfp);
+
+ 
+        // Generate an error if c5spd is > 512k, or samplelength > 256 megs
+        //  (nothing would EVER be that high)
+
+        if(feof(modfp) || (s.c5spd > 0x7ffffL) || (s.length > 0xfffffffUL) ||
+          (s.loopbeg > 0xfffffffUL) || (s.loopend > 0xfffffffUL))
+        {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+            return 0;
+        }
+
+        q->samplename = DupStr(s.sampname,26);
+
+        q->speed      = s.c5spd / 2;
+        q->panning    = ((s.panning & 127)==64) ? 255 : (s.panning & 127) << 2;
+        q->length     = s.length;
+        q->loopstart  = s.loopbeg;
+        q->loopend    = s.loopend;
+        q->volume     = s.volume;
+        q->globvol    = s.globvol;
+        q->seekpos    = s.sampoffset;
+
+        // ===================================
+        // Convert speed to XM linear finetune
+
+        if(of.flags & UF_LINEAR)
+            q->speed = cvt_c5spd_to_finetune(s.c5spd, t);
+
+        if(s.panning & 128) q->flags |= SF_OWNPAN;
+
+        if(s.vibrate)
+        {   q->vibflags |= AV_IT;
+            q->vibtype   = s.vibwave;
+            q->vibsweep  = s.vibrate * 2;
+            q->vibdepth  = s.vibdepth;
+            q->vibrate   = s.vibspeed;
+        }
+
+        if(s.flag & 2)  q->flags |= SF_16BITS;
+        if(s.flag & 16) q->flags |= SF_LOOP;
+        if(s.flag & 64) q->flags |= SF_BIDI;
+
+        if(mh->cwt >= 0x200)
+        {   if(s.convert & 1) q->flags |= SF_SIGNED;
+            if(s.convert & 4) q->flags |= SF_DELTA;   
+        }
+                              
+        q++;
+    }
+
+    // Load instruments if instrument mode flag enabled
+
+    if(mh->flags & 4)
+    {   if(!AllocInstruments()) return 0;
+        d = of.instruments;
+        of.flags |= UF_NNA | UF_INST;
+
+        for(t=0; t<mh->insnum; t++)
+        {   ITINSTHEADER ih;
+
+            // seek to instrument position
+            _mm_fseek(modfp,paraptr[t]+4,SEEK_SET);
+
+            // and load instrument info
+            _mm_read_string(ih.filename,12,modfp);
+            ih.zerobyte  = _mm_read_UBYTE(modfp);
+            if(mh->cwt < 0x200) // load IT 1.xx inst header
+            {   ih.volflg    = _mm_read_UBYTE(modfp);
+                ih.volbeg    = _mm_read_UBYTE(modfp);
+                ih.volend    = _mm_read_UBYTE(modfp);
+                ih.volsusbeg = _mm_read_UBYTE(modfp);
+                ih.volsusend = _mm_read_UBYTE(modfp);
+                _mm_read_I_UWORD(modfp);
+                ih.fadeout   = _mm_read_I_UWORD(modfp);
+                ih.nna       = _mm_read_UBYTE(modfp);
+                ih.dnc       = _mm_read_UBYTE(modfp);
+            } else   // Read IT200+ header
+            {   ih.nna       = _mm_read_UBYTE(modfp);
+                ih.dct       = _mm_read_UBYTE(modfp);
+                ih.dca       = _mm_read_UBYTE(modfp);
+                ih.fadeout   = _mm_read_I_UWORD(modfp);
+                ih.ppsep     = _mm_read_UBYTE(modfp);
+                ih.ppcenter  = _mm_read_UBYTE(modfp);
+                ih.globvol   = _mm_read_UBYTE(modfp);
+                ih.chanpan   = _mm_read_UBYTE(modfp);
+                ih.rvolvar   = _mm_read_UBYTE(modfp);
+                ih.rpanvar   = _mm_read_UBYTE(modfp);
+            }
+
+            ih.trkvers   = _mm_read_I_UWORD(modfp);
+            ih.numsmp    = _mm_read_UBYTE(modfp);
+            _mm_read_UBYTE(modfp);
+            _mm_read_string(ih.name,26,modfp);
+            _mm_read_UBYTES(ih.blank01,6,modfp);
+            _mm_read_I_UWORDS(ih.samptable,120,modfp);
+            if(mh->cwt < 0x200)  // load IT 1xx volume envelope
+            {   _mm_read_UBYTES(ih.volenv,200,modfp);
+                for(lp=0; lp<25; lp++)
+                {   ih.oldvoltick[lp] = _mm_read_UBYTE(modfp);
+                    ih.volnode[lp]    = _mm_read_UBYTE(modfp);
+                }
+            } else // load IT 2xx vol & chanpan & pitch envs
+            {   ih.volflg    = _mm_read_UBYTE(modfp);
+                ih.volpts    = _mm_read_UBYTE(modfp);
+                ih.volbeg    = _mm_read_UBYTE(modfp);
+                ih.volend    = _mm_read_UBYTE(modfp);
+                ih.volsusbeg = _mm_read_UBYTE(modfp);
+                ih.volsusend = _mm_read_UBYTE(modfp);
+                for(lp=0; lp<25; lp++)
+                {   ih.volnode[lp] = _mm_read_UBYTE(modfp);
+                    ih.voltick[lp] = _mm_read_I_UWORD(modfp);
+                }
+                _mm_read_UBYTE(modfp);
+
+                ih.panflg    = _mm_read_UBYTE(modfp);
+                ih.panpts    = _mm_read_UBYTE(modfp);
+                ih.panbeg    = _mm_read_UBYTE(modfp);
+                ih.panend    = _mm_read_UBYTE(modfp);
+                ih.pansusbeg = _mm_read_UBYTE(modfp);
+                ih.pansusend = _mm_read_UBYTE(modfp);
+                for(lp=0; lp<25; lp++)
+                {   ih.pannode[lp] = _mm_read_SBYTE(modfp);
+                    ih.pantick[lp] = _mm_read_I_UWORD(modfp);
+                }
+                _mm_read_UBYTE(modfp);
+
+                ih.pitflg    = _mm_read_UBYTE(modfp);
+                ih.pitpts    = _mm_read_UBYTE(modfp);
+                ih.pitbeg    = _mm_read_UBYTE(modfp);
+                ih.pitend    = _mm_read_UBYTE(modfp);
+                ih.pitsusbeg = _mm_read_UBYTE(modfp);
+                ih.pitsusend = _mm_read_UBYTE(modfp);
+                for(lp=0; lp<25; lp++)
+                {   ih.pitnode[lp] = _mm_read_SBYTE(modfp);
+                    ih.pittick[lp] = _mm_read_I_UWORD(modfp);
+                }
+                _mm_read_UBYTE(modfp);
+            }
+ 
+            if(feof(modfp))
+            {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+                return 0;
+            }
+
+            d->volflg |= EF_VOLENV;
+            d->insname = DupStr(ih.name,26);
+            d->nnatype = ih.nna;
+            
+            if(mh->cwt < 0x200)
+            {   d->volfade = ih.fadeout << 6;
+                if(ih.dnc)
+                {   d->dct = DCT_NOTE;
+                    d->dca = DCA_CUT;
+                }
+
+                if(ih.volflg & 1) d->volflg |= EF_ON;
+                if(ih.volflg & 2) d->volflg |= EF_LOOP;
+                if(ih.volflg & 4) d->volflg |= EF_SUSTAIN;      
+               
+                // XM conversion of IT envelope Array
+                
+                d->volbeg    = ih.volbeg;   
+                d->volend    = ih.volend;
+                d->volsusbeg = ih.volsusbeg;
+                d->volsusend = ih.volsusend;
+            
+                if(ih.volflg & 1)
+                {   for(u=0; u<25; u++)
+                        if(ih.oldvoltick[d->volpts] != 0xff)
+                        {   d->volenv[d->volpts].val = (ih.volnode[d->volpts] << 2);
+                            d->volenv[d->volpts].pos = ih.oldvoltick[d->volpts];
+                            d->volpts++;
+                        } else break;
+                }  
+            } else
+            {   d->panning = ((ih.chanpan&127) == 64) ? 255 : (ih.chanpan&127)<<2;
+                if(!(ih.chanpan & 128)) d->flags |= IF_OWNPAN;
+
+                if(!(ih.ppsep & 128))
+                {   d->pitpansep = ih.ppsep << 2;
+                    d->pitpancenter= ih.ppcenter;
+                    d->flags |= IF_PITCHPAN;
+                }
+                d->globvol = ih.globvol >> 1;
+                d->volfade = ih.fadeout << 5;
+                d->dct     = ih.dct;
+                d->dca     = ih.dca;
+
+                if(mh->cwt >= 0x204)
+                {   d->rvolvar = ih.rvolvar;
+                    d->rpanvar = ih.rpanvar;
+                }
+
+                if(ih.volflg & 1) d->volflg |= EF_ON;
+                if(ih.volflg & 2) d->volflg |= EF_LOOP;
+                if(ih.volflg & 4) d->volflg |= EF_SUSTAIN;
+
+                if(ih.panflg & 1) d->panflg |= EF_ON;
+                if(ih.panflg & 2) d->panflg |= EF_LOOP;
+                if(ih.panflg & 4) d->panflg |= EF_SUSTAIN;      
+
+                if(ih.pitflg & 1) d->pitflg |= EF_ON;
+                if(ih.pitflg & 2) d->pitflg |= EF_LOOP;
+                if(ih.pitflg & 4) d->pitflg |= EF_SUSTAIN;      
+
+                d->volpts    = ih.volpts;
+                d->volbeg    = ih.volbeg;   
+                d->volend    = ih.volend;
+                d->volsusbeg = ih.volsusbeg;
+                d->volsusend = ih.volsusend;
+
+                for(u=0; u<ih.volpts; u++)
+                {   d->volenv[u].val = (ih.volnode[u] << 2);
+                    d->volenv[u].pos = ih.voltick[u];
+                }
+
+                d->panpts    = ih.panpts;
+                d->panbeg    = ih.panbeg;
+                d->panend    = ih.panend;
+                d->pansusbeg = ih.pansusbeg;
+                d->pansusend = ih.pansusend;
+ 
+                for(u=0; u<ih.panpts; u++)                                               
+                {   d->panenv[u].val = (ih.pannode[u]+32) << 2;
+                    d->panenv[u].pos = ih.pantick[u];
+                }
+
+                d->pitpts    = ih.pitpts;
+                d->pitbeg    = ih.pitbeg;
+                d->pitend    = ih.pitend;
+                d->pitsusbeg = ih.pitsusbeg;
+                d->pitsusend = ih.pitsusend;
+
+                for(u=0; u<ih.pitpts; u++)
+                {   d->pitenv[u].val = (ih.pitnode[u]+32);
+                    d->pitenv[u].pos = ih.pittick[u];
+                }
+            }
+   
+            if(of.flags & UF_LINEAR)
+            {   for(u=0; u<120; u++)
+                {   d->samplenote[u]   = (ih.samptable[u] & 255);
+                    d->samplenumber[u] = (ih.samptable[u] >> 8) ? ((ih.samptable[u] >> 8) - 1) : 255;
+                    if(d->samplenumber[u]!=255)
+                        d->samplenote[u] += noteindex[d->samplenumber[u]];
+                }
+            } else
+            {   for(u=0; u<120; u++)
+                {   d->samplenote[u]   = (ih.samptable[u] & 255);
+                    d->samplenumber[u] = (ih.samptable[u] >> 8) ? ((ih.samptable[u] >> 8) - 1) : 255;
+                }
+            }
+
+            d++;                  
+        }
+    } else if(of.flags & UF_LINEAR)
+    {   if(!AllocInstruments()) return 0;
+        d = of.instruments;
+        of.flags |= UF_INST;
+
+        for(t=0; t<mh->smpnum; t++, d++)
+        {   for(u=0; u<120; u++)
+                d->samplenote[u] += noteindex[d->samplenumber[u]];
+        }
+    }
+
+
+    // Figure out how many channels this blasted song actually uses (what
+    // ever happened to common courtesy of storing this simple value
+    // somewhere in the damn module, eh!?)
+
+    of.numchn = 0;
+    memset(remap,-1,64*sizeof(UBYTE));
+    
+    for(t=0; t<of.numpat; t++)
+    {   UWORD packlen;
+
+        // seek to pattern position
+        if(paraptr[mh->insnum+mh->smpnum+t] != 0)  // No parapointer = pattern of 64 rows, EMPTY
+        {   _mm_fseek(modfp,(((long)paraptr[mh->insnum+mh->smpnum+t])),SEEK_SET);
+            packlen = _mm_read_I_UWORD(modfp);
+            packlen = _mm_read_I_UWORD(modfp);    // read pattern length (# of rows)
+            _mm_read_I_ULONG(modfp);
+            if(IT_GetNumChannels(packlen)) return 0;
+        }
+    }
+
+    // give each of them a different number
+    for(t=0; t<64; t++)
+    {   if(remap[t]==0)
+        {   remap[t] = of.numchn;
+            of.numchn++;
+        }
+    }
+
+    of.numtrk = of.numpat*of.numchn;
+
+
+    if(!AllocPatterns()) return 0;
+    if(!AllocTracks()) return 0;
+
+    for(t=0; t<of.numpat; t++)
+    {   UWORD packlen;
+
+        // seek to pattern position
+        if(paraptr[mh->insnum+mh->smpnum+t] == 0)  // No parapointer = pattern of 64 rows, EMPTY
+        {  of.pattrows[t] = 64;
+           for(u=0; u<of.numchn; u++)
+           {  int k;
+              UniReset();
+              for(k=0; k<64; k++) UniNewline();
+              of.tracks[numtrk++] = UniDup();
+           }
+        } else
+        {  _mm_fseek(modfp,(((long)paraptr[mh->insnum+mh->smpnum+t])),SEEK_SET);
+           packlen = _mm_read_I_UWORD(modfp);
+           of.pattrows[t] = _mm_read_I_UWORD(modfp);
+           _mm_read_I_ULONG(modfp);
+
+           if(!IT_ReadPattern(of.pattrows[t])) return 0;
+        }
+    }
+
+    return 1;
+}
+
+
+CHAR *IT_LoadTitle(void)
+{
+   CHAR s[26];
+
+   _mm_fseek(modfp,4,SEEK_SET);
+   if(!fread(s,26,1,modfp)) return NULL;
+   
+   return(DupStr(s,26));
+}
+
+
+MLOADER load_it =
+{   NULL,
+    "IT",
+    "Portable IT loader v0.2",
+    IT_Init,
+    IT_Test,
+    IT_Load,
+    IT_Cleanup,
+
+    IT_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_m15.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,410 @@
+/*
+
+ Name: LOAD_M15.C
+
+ Description:
+ 15 instrument MOD loader
+ Also supports Ultimate Sound Tracker (old M15 format)
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/*************************************************************************
+*************************************************************************/
+
+
+typedef struct MSAMPINFO       // sample header as it appears in a module
+{   CHAR  samplename[22];
+    UWORD length;
+    UBYTE finetune;
+    UBYTE volume;
+    UWORD reppos;
+    UWORD replen;
+} MSAMPINFO;
+
+
+typedef struct MODULEHEADER          // verbatim module header
+{   CHAR       songname[20];         // the songname..
+    MSAMPINFO  samples[15];          // all sampleinfo
+    UBYTE      songlength;           // number of patterns used
+    UBYTE      magic1;               // should be 127
+    UBYTE      positions[128];       // which pattern to play at pos
+} MODULEHEADER;
+
+
+typedef struct MODNOTE
+{   UBYTE a,b,c,d;
+} MODNOTE;
+
+
+/*************************************************************************
+*************************************************************************/
+
+static MODULEHEADER *mh = NULL;       // raw as-is module header
+static MODNOTE *patbuf = NULL;
+static BOOL ust_loader = 0;          // if TRUE, load as a ust module.
+static CHAR nulls[3] = {0,0,0};
+
+static BOOL LoadModuleHeader(MODULEHEADER *mh)
+{
+    int t;
+
+    _mm_read_string(mh->songname,20,modfp);
+
+    for(t=0; t<15; t++)
+    {   MSAMPINFO *s = &mh->samples[t];
+        _mm_read_string(s->samplename,22,modfp);
+        s->length   =_mm_read_M_UWORD(modfp);
+        s->finetune =_mm_read_UBYTE(modfp);
+        s->volume   =_mm_read_UBYTE(modfp);
+        s->reppos   =_mm_read_M_UWORD(modfp);
+        s->replen   =_mm_read_M_UWORD(modfp);
+    }
+
+    mh->songlength  =_mm_read_UBYTE(modfp);
+    mh->magic1      =_mm_read_UBYTE(modfp);               // should be 127
+    _mm_read_UBYTES(mh->positions,128,modfp);
+
+    return(!feof(modfp));
+}
+
+
+static int CheckPatternType(int numpat)
+
+// Checks the patterns in the modfile for UST / 15-inst indications.
+// For example, if an effect 3xx is found, it is assumed that the song
+// is 15-inst.  If a 1xx effect has dat greater than 0x20, it is UST.
+//  Returns:  0 indecisive; 1 = UST; 2 = 15-inst
+
+{
+    int   t;
+    UBYTE eff, dat;
+
+    ust_loader = 1;
+
+    for(t=0; t<numpat*(64U*4); t++)
+    {   // Load the pattern into the temp buffer
+        // and convert it
+
+        _mm_read_UBYTE(modfp);      // read note
+        _mm_read_UBYTE(modfp);      // read inst
+        eff = _mm_read_UBYTE(modfp);
+        dat = _mm_read_UBYTE(modfp);
+        
+        if((eff==3) && (dat!=0) || (eff >= 2))   return 2;
+        if(eff==1)
+        {   if(dat > 0x1f) return 1;
+            if(dat < 0x3)  return 2;
+        }
+        if((eff==2) && (dat > 0x1f)) return 1;
+    }
+
+    return 0;
+}
+
+
+BOOL M15_Test(void)
+{
+    int          t, numpat;
+    MODULEHEADER mh;
+
+    ust_loader = 0;
+
+    if(!LoadModuleHeader(&mh)) return 0;
+    if(mh.magic1>127) return 0;
+
+    for(t=0; t<15; t++)
+    {   // all finetunes should be zero
+        if(mh.samples[t].finetune != 0) return 0;
+
+        // all volumes should be <= 64
+        if(mh.samples[t].volume > 64) return 0;
+
+        // all instrument names should begin with s, st-, or a number
+        if(mh.samples[t].samplename[0] == 's')
+        {   if((memcmp(mh.samples[t].samplename,"st-",3) != 0) &&
+               (memcmp(mh.samples[t].samplename,"ST-",3) != 0) &&
+               (memcmp(mh.samples[t].samplename,nulls,3) != 0))
+                ust_loader = 1;
+        } else if((mh.samples[t].samplename[0] < '0') || (mh.samples[t].samplename[0] > '9'))
+            ust_loader = 1;
+
+        if(mh.samples[t].length > 4999)
+        {   ust_loader = 0;
+            if(mh.samples[t].length > 32768) return 0;
+        }
+
+        if(!ust_loader) return 1;
+
+        if(((mh.samples[t].reppos) + mh.samples[t].replen) > (mh.samples[t].length + 10))
+        {   ust_loader = 1;
+            return 1;
+        }
+        
+    }
+
+    for(numpat=0, t=0; t<mh.songlength; t++)
+    {   if(mh.positions[t] > numpat)
+            numpat = mh.positions[t];
+    }
+
+    numpat++;
+    switch(CheckPatternType(numpat))
+    {   case 0:   // indecisive, so check more clues...
+
+        break;
+
+        case 1:  ust_loader = 1;   break;
+        case 2:  ust_loader = 0;   break;
+    }
+ 
+    return 1;
+}
+
+
+BOOL M15_Init(void)
+{
+    if(!(mh=(MODULEHEADER *)_mm_calloc(1,sizeof(MODULEHEADER)))) return 0;
+    return 1;
+}
+
+
+void M15_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    if(patbuf!=NULL) free(patbuf);
+
+    mh = NULL;
+    patbuf = NULL;
+}
+
+
+/*
+Old (amiga) noteinfo:
+
+ _____byte 1_____   byte2_    _____byte 3_____   byte4_
+/                \ /      \  /                \ /      \
+0000          0000-00000000  0000          0000-00000000
+
+Upper four    12 bits for    Lower four    Effect command.
+bits of sam-  note period.   bits of sam-
+ple number.                  ple number.
+
+*/
+
+
+static void M15_ConvertNote(MODNOTE *n)
+{
+    UBYTE instrument,effect,effdat,note;
+    UWORD period;
+
+    // extract the various information from the 4 bytes that
+    // make up a single note
+
+    instrument = (n->a&0x10)|(n->c>>4);
+    period     = (((UWORD)n->a&0xf)<<8)+n->b;
+    effect     = n->c&0xf;
+    effdat     = n->d;
+
+    // Convert the period to a note number
+
+    note=0;
+    if(period != 0)
+    {   for(note=0; note<60; note++)
+            if(period >= npertab[note]) break;
+        note++;
+        if(note==61) note = 0;
+    }
+
+    if(instrument!=0) UniInstrument(instrument-1);
+    if(note!=0) UniNote(note+23);
+
+    // Convert pattern jump from Dec to Hex
+    if(effect == 0xd)
+        effdat = (((effdat&0xf0)>>4)*10)+(effdat&0xf);
+
+    if(ust_loader)
+    {   switch(effect)
+        {   case 0:  break;
+            case 1:
+                UniPTEffect(0,effdat);
+            break;
+
+            case 2:  
+                if(effdat&0xf) UniPTEffect(1,effdat&0xf);
+                if(effdat>>2)  UniPTEffect(2,effdat>>2);
+            break;
+
+            case 3:  break;
+
+            default:
+                UniPTEffect(effect,effdat);
+            break;
+        }
+    } else UniPTEffect(effect,effdat);
+}
+
+
+static UBYTE *M15_ConvertTrack(MODNOTE *n)
+{
+    int t;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {   M15_ConvertNote(n);
+        UniNewline();
+        n += 4;
+    }
+    return UniDup();
+}
+
+
+
+static BOOL M15_LoadPatterns(void)
+//  Loads all patterns of a modfile and converts them into the
+//  3 byte format.
+{
+    int t,s,tracks=0;
+
+    if(!AllocPatterns()) return 0;
+    if(!AllocTracks()) return 0;
+
+    // Allocate temporary buffer for loading
+    // and converting the patterns
+
+    if(!(patbuf=(MODNOTE *)_mm_calloc(64U*4,sizeof(MODNOTE)))) return 0;
+
+    for(t=0; t<of.numpat; t++)
+    {   // Load the pattern into the temp buffer
+        // and convert it
+
+        for(s=0; s<(64U*4); s++)
+        {   patbuf[s].a=_mm_read_UBYTE(modfp);
+            patbuf[s].b=_mm_read_UBYTE(modfp);
+            patbuf[s].c=_mm_read_UBYTE(modfp);
+            patbuf[s].d=_mm_read_UBYTE(modfp);
+        }
+
+        for(s=0; s<4; s++)
+            if(!(of.tracks[tracks++]=M15_ConvertTrack(patbuf+s))) return 0;
+    }
+
+    return 1;
+}
+
+
+BOOL M15_Load(void)
+{
+    int        t;
+    SAMPLE     *q;
+    MSAMPINFO  *s;           // old module sampleinfo
+
+    // try to read module header
+
+    if(!LoadModuleHeader(mh))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+
+    if(ust_loader)
+        of.modtype = strdup("Ultimate Soundtracker");
+    else
+        of.modtype = strdup("Soundtracker");
+
+    // set module variables
+
+    of.initspeed = 6;
+    of.inittempo = 125;
+    of.numchn    = 4;                           // get number of channels
+    of.songname  = DupStr(mh->songname,20);     // make a cstr of songname
+    of.numpos    = mh->songlength;              // copy the songlength
+
+    if(!AllocPositions(of.numpos)) return 0;
+    for(t=0; t<of.numpos; t++)
+        of.positions[t] = mh->positions[t];
+
+
+    // Count the number of patterns
+
+    of.numpat = 0;
+
+    for(t=0; t<of.numpos; t++)
+    {   if(of.positions[t] > of.numpat)
+            of.numpat = of.positions[t];
+    }
+    of.numpat++;
+    of.numtrk = of.numpat*4;
+
+    // Finally, init the sampleinfo structures
+
+    of.numins = of.numsmp = 15;
+    if(!AllocSamples()) return 0;
+
+    s = mh->samples;          // init source pointer
+    q = of.samples;
+    
+    for(t=0; t<of.numins; t++)
+    {   // convert the samplename
+        q->samplename = DupStr(s->samplename,22);
+
+        // init the sampleinfo variables and
+        // convert the size pointers to longword format
+
+        q->speed     = finetune[s->finetune&0xf];
+        q->volume    = s->volume;
+        if(ust_loader)
+            q->loopstart = s->reppos;
+        else
+            q->loopstart = s->reppos<<1;
+        q->loopend   = q->loopstart+(s->replen<<1);
+        q->length    = s->length<<1;
+
+        q->flags = SF_SIGNED | SF_UST_LOOP;
+        if(s->replen>1) q->flags |= SF_LOOP;
+
+        // fix replen if repend>length
+
+        if(q->loopend>q->length) q->loopend = q->length;
+
+        s++;    // point to next source sampleinfo
+        q++;
+    }
+
+    if(!M15_LoadPatterns()) return 0;
+
+    ust_loader = 0;
+    return 1;
+}
+
+
+CHAR *M15_LoadTitle(void)
+{
+   CHAR s[20];
+
+   fseek(modfp,0,SEEK_SET);
+   if(!fread(s,22,1,modfp)) return NULL;
+
+   return(DupStr(s,20));
+}
+
+
+MLOADER load_m15 =
+{   NULL,
+    "15-instrument module",
+    "Portable MOD-15 loader v0.1",
+    M15_Init,
+    M15_Test,
+    M15_Load,
+    M15_Cleanup,
+    M15_LoadTitle
+};
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_med.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,490 @@
+/*
+
+ Name: LOAD_MED.C
+
+ Description:
+ Amiga MED module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+#define MMD0_string 0x4D4D4430
+#define MMD1_string 0x4D4D4431
+
+typedef struct MMD0
+{   ULONG   id;
+    ULONG   modlen;
+    ULONG   MMD0songP;        // struct MMD0song *song;
+    UWORD   psecnum;          // for the player routine, MMD2 only
+    UWORD   pseq;             //  "   "   "   "
+    ULONG   MMD0BlockPP;      // struct MMD0Block **blockarr;
+    ULONG   reserved1;
+    ULONG   InstrHdrPP;       // struct InstrHdr **smplarr;
+    ULONG   reserved2;
+    ULONG   MMD0expP;         // struct MMD0exp *expdata;
+    ULONG   reserved3;
+    UWORD   pstate;           // some data for the player routine
+    UWORD   pblock;
+    UWORD   pline;
+    UWORD   pseqnum;
+    SWORD   actplayline;
+    UBYTE   counter;
+    UBYTE   extra_songs;      // number of songs - 1
+} MMD0;
+
+
+typedef struct MMD0sample
+{   UWORD rep,replen;        // offs: 0(s), 2(s)
+    UBYTE midich;            // offs: 4(s)
+    UBYTE midipreset;        // offs: 5(s)
+    UBYTE svol;              // offs: 6(s)
+    SBYTE strans;            // offs: 7(s)
+} MMD0sample;
+
+
+typedef struct MMD0song
+{   MMD0sample sample[63];   // 63 * 8 bytes = 504 bytes
+    UWORD   numblocks;       // offs: 504
+    UWORD   songlen;         // offs: 506
+    UBYTE   playseq[256];    // offs: 508
+    UWORD   deftempo;        // offs: 764
+    SBYTE   playtransp;      // offs: 766
+    UBYTE   flags;           // offs: 767
+    UBYTE   flags2;          // offs: 768
+    UBYTE   tempo2;          // offs: 769
+    UBYTE   trkvol[16];      // offs: 770
+    UBYTE   mastervol;       // offs: 786
+    UBYTE   numsamples;      // offs: 787
+} MMD0song;
+
+
+typedef struct MMD0NOTE
+{  UBYTE a,b,c;
+} MMD0NOTE;
+
+
+typedef struct MMD1NOTE
+{   UBYTE a,b,c,d;
+} MMD1NOTE;
+
+
+typedef struct InstrHdr
+{   ULONG   length;
+    SWORD   type;
+    // Followed by actual data
+} InstrHdr;
+
+
+static MMD0 *mh     = NULL;
+static MMD0song *ms = NULL;
+static ULONG *ba    = NULL;
+static MMD0NOTE *mmd0pat = NULL;
+static MMD1NOTE *mmd1pat = NULL;
+
+#define d0note(row,col) mmd0pat[(row*(UWORD)of.numchn)+col]
+#define d1note(row,col) mmd1pat[(row*(UWORD)of.numchn)+col]
+
+
+static CHAR MED_Version[] = "MED";
+
+
+BOOL MED_Test(void)
+{
+    UBYTE id[4];
+
+    if(!_mm_read_UBYTES(id,4,modfp)) return 0;
+    if(!memcmp(id,"MMD0",4)) return 1;
+    if(!memcmp(id,"MMD1",4)) return 1;
+    return 0;
+}
+
+
+BOOL MED_Init(void)
+{
+    if(!(mh=(MMD0 *)_mm_calloc(1,sizeof(MMD0)))) return 0;
+    if(!(ms=(MMD0song *)_mm_calloc(1,sizeof(MMD0song)))) return 0;
+    return 1;
+}
+
+
+void MED_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    if(ms!=NULL) free(ms);
+    if(ba!=NULL) free(ba);
+    if(mmd0pat!=NULL) free(mmd0pat);
+    if(mmd1pat!=NULL) free(mmd1pat);
+
+    mh = NULL;
+    ms = NULL;
+    ba = NULL;        // blockarr
+    mmd0pat = NULL;
+    mmd1pat = NULL;    
+}
+
+
+void EffectCvt(UBYTE eff,UBYTE dat)
+{
+    switch(eff)
+    {   // 0x0 0x1 0x2 0x3 0x4      // PT effects
+        case 0x5:       // PT vibrato with speed/depth nibbles swapped
+           UniPTEffect(0x4,(dat>>4) | ((dat&0xf)<<4) );
+        break;
+
+        case 0x6:       // not used
+        case 0x7:       // not used
+        case 0x8:       // midi hold/decay
+        break;
+
+        case 0x9:
+           if(dat<=0x20) UniPTEffect(0xf,dat);
+        break;
+
+        // 0xa 0xb 0xc all PT effects
+
+        case 0xd:       // same as PT volslide
+           UniPTEffect(0xa,dat);
+        break;
+
+        case 0xe:       // synth jmp - midi
+        break;
+
+        case 0xf:
+           // F00 does patternbreak with med
+           if(dat==0) UniPTEffect(0xd,0);
+           else if(dat<=0xa) UniPTEffect(0xf,dat);
+           else if(dat<0xf1) UniPTEffect(0xf,((UWORD)dat*125)/33);
+           else if(dat==0xff) UniPTEffect(0xc,0);  // stop note
+        break;
+
+        default:        // all normal PT effects are handled here :)
+           // Convert pattern jump from Dec to Hex
+           if(eff == 0xd)
+               dat = (((dat&0xf0)>>4)*10)+(dat&0xf);
+           UniPTEffect(eff,dat);
+        break;
+    }
+}
+
+
+
+UBYTE *MED_Convert1(int col)
+{
+    int t;
+    UBYTE a,b,c,d,inst,note,eff,dat;
+    MMD1NOTE *n;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {   n = &d1note(t,col);
+        a = n->a;
+        b = n->b;
+        c = n->c;
+        d = n->d;
+
+        note = a&0x7f;
+        inst = b&0x3f;
+        eff  = c&0xf;
+        dat  = d;
+
+        if(inst!=0) UniInstrument(inst-1);
+        if(note!=0) UniNote(note+23);
+
+        EffectCvt(eff,dat);
+        UniNewline();
+    }
+
+    return UniDup();
+}
+
+
+UBYTE *MED_Convert0(int col)
+{
+    int t;
+    UBYTE a,b,c,inst,note,eff,dat;
+    MMD0NOTE *n;
+
+    UniReset();
+    for(t=0;t<64;t++)
+    {   n = &d0note(t,col);
+        a = n->a;
+        b = n->b;
+        c = n->c;
+
+        note = a & 0x3f;
+        a  >>= 6;
+        a    = ((a & 1) << 1) | (a >> 1);
+
+        inst = (b >> 4) | (a << 4);
+        eff  = b & 0xf;
+        dat  = c;
+
+        if(inst!=0) UniInstrument(inst-1);
+        if(note!=0) UniNote(note+35);
+
+        EffectCvt(eff,dat);
+        UniNewline();
+    }
+    return UniDup();
+}
+
+
+BOOL LoadMMD0Patterns(void)
+{
+   int      t,row,col;
+   UWORD    numtracks,numlines,maxlines=0,track=0;
+   MMD0NOTE *mmdp;
+   
+   // first, scan patterns to see how many channels are used
+   for(t=0; t<of.numpat; t++)
+   {   _mm_fseek(modfp,ba[t],SEEK_SET);
+       numtracks = _mm_read_UBYTE(modfp);
+       numlines  = _mm_read_UBYTE(modfp);
+
+       if(numtracks>of.numchn) of.numchn = numtracks;
+       if(numlines>maxlines) maxlines = numlines;
+   }
+
+   of.numtrk = of.numpat*of.numchn;
+   if(!AllocTracks()) return 0;
+   if(!AllocPatterns()) return 0;
+
+   if(!(mmd0pat=(MMD0NOTE*)_mm_calloc(of.numchn*(maxlines+1),sizeof(MMD0NOTE)))) return 0;
+
+   // second read: no more mr. nice guy,
+   // really read and convert patterns
+
+   for(t=0; t<of.numpat; t++)
+   {   _mm_fseek(modfp,ba[t],SEEK_SET);
+       numtracks = _mm_read_UBYTE(modfp);
+       numlines  = _mm_read_UBYTE(modfp);
+
+       of.pattrows[t] = numlines+1;
+       memset(mmdp=mmd0pat,0,of.numchn*maxlines*sizeof(MMD0NOTE));
+       for(row=numlines+1; row; row--)
+       {   for(col=numtracks; col; col--,mmdp++)
+           {   mmdp->a = _mm_read_UBYTE(modfp);
+               mmdp->b = _mm_read_UBYTE(modfp);
+               mmdp->c = _mm_read_UBYTE(modfp);
+           }
+       }
+
+       for(col=0; col<of.numchn; col++)
+       {   of.tracks[track] = MED_Convert0(col);
+           track++;
+       }
+    }
+    return 1;
+}
+
+
+BOOL LoadMMD1Patterns(void)
+{
+    int      t,row,col;
+    UWORD    numtracks,numlines,maxlines=0,track=0;
+    MMD1NOTE *mmdp;
+
+    // first, scan patterns to see how many channels are used
+    for(t=0; t<of.numpat; t++)
+    {  _mm_fseek(modfp,ba[t],SEEK_SET);
+       numtracks = _mm_read_M_UWORD(modfp);
+       numlines  = _mm_read_M_UWORD(modfp);
+
+       _mm_fseek(modfp,sizeof(ULONG),SEEK_CUR);
+       if(numtracks>of.numchn) of.numchn = numtracks;
+       if(numlines>maxlines) maxlines = numlines;
+    }
+
+    of.numtrk = of.numpat*of.numchn;
+    if(!AllocTracks()) return 0;
+    if(!AllocPatterns()) return 0;
+
+    if(!(mmd1pat=(MMD1NOTE*)_mm_calloc(of.numchn*(maxlines+1),sizeof(MMD1NOTE)))) return 0;
+
+    // second read: no more mr. nice guy, really read and convert patterns
+    for(t=0; t<of.numpat; t++)
+    {   _mm_fseek(modfp,ba[t],SEEK_SET);
+        numtracks = _mm_read_M_UWORD(modfp);
+        numlines  = _mm_read_M_UWORD(modfp);
+
+        _mm_fseek(modfp,sizeof(ULONG),SEEK_CUR);
+        of.pattrows[t] = numlines;
+        memset(mmdp=mmd1pat,0,of.numchn*maxlines*sizeof(MMD1NOTE));
+
+        for(row=numlines+1; row; row--)
+        {   for(col=numtracks; col; col--,mmdp++)
+            {   mmdp->a = _mm_read_UBYTE(modfp);
+                mmdp->b = _mm_read_UBYTE(modfp);
+                mmdp->c = _mm_read_UBYTE(modfp);
+                mmdp->d = _mm_read_UBYTE(modfp);
+            }
+        }
+
+        for(col=0;col<of.numchn;col++)
+        {   of.tracks[track]=MED_Convert1(col);
+            track++;
+        }
+    }
+    return 1;
+}
+
+
+
+BOOL MED_Load(void)
+{
+    int        t;
+    ULONG      sa[64];
+    InstrHdr   s;
+    SAMPLE     *q;
+    MMD0sample *mss;
+
+    // try to read module header
+
+    mh->id          = _mm_read_M_ULONG(modfp);
+    mh->modlen      = _mm_read_M_ULONG(modfp);
+    mh->MMD0songP   = _mm_read_M_ULONG(modfp);
+    mh->psecnum     = _mm_read_M_UWORD(modfp);
+    mh->pseq        = _mm_read_M_UWORD(modfp);
+    mh->MMD0BlockPP = _mm_read_M_ULONG(modfp);
+    mh->reserved1   = _mm_read_M_ULONG(modfp);
+    mh->InstrHdrPP  = _mm_read_M_ULONG(modfp);
+    mh->reserved2   = _mm_read_M_ULONG(modfp);
+    mh->MMD0expP    = _mm_read_M_ULONG(modfp);
+    mh->reserved3   = _mm_read_M_ULONG(modfp);
+    mh->pstate      = _mm_read_M_UWORD(modfp);
+    mh->pblock      = _mm_read_M_UWORD(modfp);
+    mh->pline       = _mm_read_M_UWORD(modfp);
+    mh->pseqnum     = _mm_read_M_UWORD(modfp);
+    mh->actplayline = _mm_read_M_SWORD(modfp);
+    mh->counter     = _mm_read_UBYTE(modfp);
+    mh->extra_songs = _mm_read_UBYTE(modfp);
+
+    // Seek to MMD0song struct
+    _mm_fseek(modfp,mh->MMD0songP,SEEK_SET);
+
+
+    // Load the MMD0 Song Header
+
+    mss = ms->sample;     // load the sample data first
+    for(t=63; t; t--, mss++)
+    {   mss->rep        = _mm_read_M_UWORD(modfp);
+        mss->replen     = _mm_read_M_UWORD(modfp);
+        mss->midich     = _mm_read_UBYTE(modfp);
+        mss->midipreset = _mm_read_UBYTE(modfp);
+        mss->svol       = _mm_read_UBYTE(modfp);
+        mss->strans     = _mm_read_SBYTE(modfp);
+    }
+
+    ms->numblocks  = _mm_read_M_UWORD(modfp);
+    ms->songlen    = _mm_read_M_UWORD(modfp);
+    _mm_read_UBYTES(ms->playseq,256,modfp);
+    ms->deftempo   = _mm_read_M_UWORD(modfp);
+    ms->playtransp = _mm_read_SBYTE(modfp);
+    ms->flags      = _mm_read_UBYTE(modfp);
+    ms->flags2     = _mm_read_UBYTE(modfp);
+    ms->tempo2     = _mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(ms->trkvol,16,modfp);
+    ms->mastervol  = _mm_read_UBYTE(modfp);
+    ms->numsamples = _mm_read_UBYTE(modfp);
+
+    // check for a bad header
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // seek to and read the samplepointer array
+    _mm_fseek(modfp,mh->InstrHdrPP,SEEK_SET);
+    if(!_mm_read_M_ULONGS(sa,ms->numsamples,modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // alloc and read the blockpointer array
+    if(!(ba=(ULONG *)_mm_calloc(ms->numblocks, sizeof(ULONG)))) return 0;
+    _mm_fseek(modfp,mh->MMD0BlockPP,SEEK_SET);
+    if(!_mm_read_M_ULONGS(ba,ms->numblocks,modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+
+    // copy song positions
+    if(!AllocPositions(ms->songlen)) return 0;
+    for(t=0; t<ms->songlen; t++)
+       of.positions[t] = ms->playseq[t];
+
+    of.initspeed = 6;
+    of.inittempo = ((UWORD)ms->deftempo*125)/33;
+    of.modtype   = strdup(MED_Version);
+    of.numchn    = 0;                         // will be counted later
+    of.numpat    = ms->numblocks;
+    of.numpos    = ms->songlen;
+    of.numins    = ms->numsamples;
+
+    of.numsmp = of.numins;
+    if(!AllocSamples()) return 0;
+    q = of.samples;
+   
+    for(t=0; t<of.numins; t++)
+    {   _mm_fseek(modfp,sa[t],SEEK_SET);
+        s.length = _mm_read_M_ULONG(modfp);
+        s.type   = _mm_read_M_SWORD(modfp);
+       
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+            return 0;
+        }
+
+        q->samplename = NULL;
+        q->length    = s.length;
+        q->seekpos   = _mm_ftell(modfp);
+        q->loopstart = ms->sample[t].rep<<1;
+        q->loopend   = q->loopstart+(ms->sample[t].replen<<1);
+        q->flags     = SF_SIGNED;
+        q->speed     = 8363;
+        q->volume    = 64;
+
+        if(ms->sample[t].replen>1) q->flags|=SF_LOOP;
+
+        // don't load sample if length>='MMD0' hah.. hah.. very funny.. NOT!
+        if(q->length >= MMD0_string) q->length = 0;
+
+        q++;
+    }
+
+
+    if(mh->id==MMD0_string)
+    {   if(!LoadMMD0Patterns()) return 0;
+    } else if(mh->id==MMD1_string)
+    {   if(!LoadMMD1Patterns()) return 0;
+    } else
+    {   _mm_errno = MMERR_NOT_A_MODULE;
+        return 0;
+    }
+   
+    return 1;
+}
+
+
+MLOADER load_med =
+{   NULL,
+    "MED",
+    "MED loader v0.1",
+    MED_Init,
+    MED_Test,
+    MED_Load,
+    MED_Cleanup,
+    NULL
+};
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_mod.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,341 @@
+/*
+
+ Name: LOAD_MOD.C
+
+ Description:
+ Generic MOD loader (Protracker, StarTracker, FastTracker, etc)
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/*************************************************************************
+*************************************************************************/
+
+
+typedef struct MSAMPINFO                  // sample header as it appears in a module
+{   CHAR  samplename[22];
+    UWORD length;
+    UBYTE finetune;
+    UBYTE volume;
+    UWORD reppos;
+    UWORD replen;
+} MSAMPINFO;
+
+
+typedef struct MODULEHEADER              // verbatim module header
+{   CHAR       songname[20];             // the songname..
+    MSAMPINFO  samples[31];              // all sampleinfo
+    UBYTE      songlength;               // number of patterns used
+    UBYTE      magic1;                   // should be 127
+    UBYTE      positions[128];           // which pattern to play at pos
+    UBYTE      magic2[4];                // string "M.K." or "FLT4" or "FLT8"
+} MODULEHEADER;
+
+#define MODULEHEADERSIZE 1084
+
+
+typedef struct MODTYPE              // struct to identify type of module
+{   CHAR    id[5];
+    UBYTE   channels;
+    CHAR    *name;
+} MODTYPE;
+
+
+typedef struct MODNOTE
+{   UBYTE a,b,c,d;
+} MODNOTE;
+
+
+/*************************************************************************
+*************************************************************************/
+
+
+CHAR protracker[]   = "Protracker";
+CHAR startracker[]  = "Startracker";
+CHAR fasttracker[]  = "Fasttracker";
+CHAR ins15tracker[] = "15-instrument";
+CHAR oktalyzer[]    = "Oktalyzer";
+CHAR taketracker[]  = "TakeTracker";
+
+
+MODTYPE modtypes[] =
+{   "M.K.",4,protracker,      // protracker 4 channel
+    "M!K!",4,protracker,      // protracker 4 channel
+    "FLT4",4,startracker,     // startracker 4 channel
+    "2CHN",2,fasttracker,     // fasttracker 2 channel
+    "4CHN",4,fasttracker,     // fasttracker 4 channel
+    "6CHN",6,fasttracker,     // fasttracker 6 channel
+    "8CHN",8,fasttracker,     // fasttracker 8 channel
+    "10CH",10,fasttracker,    // fasttracker 10 channel
+    "12CH",12,fasttracker,    // fasttracker 12 channel
+    "14CH",14,fasttracker,    // fasttracker 14 channel
+    "16CH",16,fasttracker,    // fasttracker 16 channel
+    "18CH",18,fasttracker,    // fasttracker 18 channel
+    "20CH",20,fasttracker,    // fasttracker 20 channel
+    "22CH",22,fasttracker,    // fasttracker 22 channel
+    "24CH",24,fasttracker,    // fasttracker 24 channel
+    "26CH",26,fasttracker,    // fasttracker 26 channel
+    "28CH",28,fasttracker,    // fasttracker 28 channel
+    "30CH",30,fasttracker,    // fasttracker 30 channel
+    "32CH",32,fasttracker,    // fasttracker 32 channel
+    "CD81",8,oktalyzer,       // atari oktalyzer 8 channel
+    "OKTA",8,oktalyzer,       // atari oktalyzer 8 channel
+    "16CN",16,taketracker,    // taketracker 16 channel
+    "32CN",32,taketracker,    // taketracker 32 channel
+    "    ",4,ins15tracker     // 15-instrument 4 channel    
+};
+
+static MODULEHEADER *mh = NULL;        // raw as-is module header
+static MODNOTE *patbuf = NULL;
+static int modtype = 0;
+
+BOOL MOD_Test(void)
+{
+    UBYTE id[4];
+
+    _mm_fseek(modfp,MODULEHEADERSIZE-4,SEEK_SET);
+    if(!fread(id,4,1,modfp)) return 0;
+
+    // find out which ID string
+
+    for(modtype=0; modtype<23; modtype++)
+        if(!memcmp(id,modtypes[modtype].id,4)) return 1;
+
+    return 0;
+}
+
+
+BOOL MOD_Init(void)
+{
+    if(!(mh=(MODULEHEADER *)_mm_calloc(1,sizeof(MODULEHEADER)))) return 0;
+    return 1;
+}
+
+
+void MOD_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    if(patbuf!=NULL) free(patbuf);
+
+    mh = NULL;
+    patbuf = NULL;
+}
+
+
+/*
+Old (amiga) noteinfo:
+
+ _____byte 1_____   byte2_    _____byte 3_____   byte4_
+/                \ /      \  /                \ /      \
+0000          0000-00000000  0000          0000-00000000
+
+Upper four    12 bits for    Lower four    Effect command.
+bits of sam-  note period.   bits of sam-
+ple number.                  ple number.
+
+*/
+
+
+void ConvertNote(MODNOTE *n)
+{
+    UBYTE instrument,effect,effdat,note;
+    UWORD period;
+
+    // extract the various information from the 4 bytes that
+    // make up a single note
+
+    instrument = (n->a&0x10)|(n->c>>4);
+    period     = (((UWORD)n->a&0xf)<<8)+n->b;
+    effect     = n->c&0xf;
+    effdat     = n->d;
+
+    // Convert the period to a note number
+
+    note=0;
+    if(period!=0)
+    {   for(note=0; note<60; note++)
+            if(period >= npertab[note]) break;
+        note++;
+        if(note==61) note = 0;
+    }
+
+    if(instrument!=0) UniInstrument(instrument-1);
+    if(note!=0) UniNote(note+23);
+
+    // Convert pattern jump from Dec to Hex
+    if(effect == 0xd)
+        effdat = (((effdat&0xf0)>>4)*10)+(effdat&0xf);
+
+    UniPTEffect(effect,effdat);
+}
+
+
+UBYTE *ConvertTrack(MODNOTE *n)
+{
+    int t;
+
+    UniReset();
+    for(t=0;t<64;t++)
+    {   ConvertNote(n);
+        UniNewline();
+        n+=of.numchn;
+    }
+    return UniDup();
+}
+
+
+BOOL ML_LoadPatterns(void)
+//  Loads all patterns of a modfile and converts them into the
+//  3 byte format.
+{
+    int t,s,tracks = 0;
+
+    if(!AllocPatterns()) return 0;
+    if(!AllocTracks())   return 0;
+
+    // Allocate temporary buffer for loading
+    // and converting the patterns
+
+    if(!(patbuf=(MODNOTE *)_mm_calloc(64U*of.numchn,sizeof(MODNOTE)))) return 0;
+
+    for(t=0; t<of.numpat; t++)
+    {   // Load the pattern into the temp buffer
+        // and convert it
+
+        for(s=0; s<(64U*of.numchn); s++)
+        {   patbuf[s].a = _mm_read_UBYTE(modfp);
+            patbuf[s].b = _mm_read_UBYTE(modfp);
+            patbuf[s].c = _mm_read_UBYTE(modfp);
+            patbuf[s].d = _mm_read_UBYTE(modfp);
+        }
+
+        for(s=0; s<of.numchn; s++)
+            if(!(of.tracks[tracks++]=ConvertTrack(patbuf+s))) return 0;
+    }
+
+    return 1;
+}
+
+
+BOOL MOD_Load(void)
+{
+    int       t;
+    SAMPLE    *q;
+    MSAMPINFO *s;           // old module sampleinfo
+
+    // try to read module header
+
+    _mm_read_string((CHAR *)mh->songname,20,modfp);
+
+    for(t=0; t<31; t++)
+    {   s = &mh->samples[t];
+        _mm_read_string(s->samplename,22,modfp);
+        s->length   =_mm_read_M_UWORD(modfp);
+        s->finetune =_mm_read_UBYTE(modfp);
+        s->volume   =_mm_read_UBYTE(modfp);
+        s->reppos   =_mm_read_M_UWORD(modfp);
+        s->replen   =_mm_read_M_UWORD(modfp);
+    }
+
+    mh->songlength  =_mm_read_UBYTE(modfp);
+    mh->magic1      =_mm_read_UBYTE(modfp);
+
+    _mm_read_UBYTES(mh->positions,128,modfp);
+    _mm_read_UBYTES(mh->magic2,4,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+
+    of.initspeed = 6;
+    of.inittempo = 125;
+    of.numchn    = modtypes[modtype].channels;      // get number of channels
+    of.modtype   = strdup(modtypes[modtype].name);  // get ascii type of mod
+    of.songname  = DupStr(mh->songname,20);         // make a cstr of songname
+    of.numpos    = mh->songlength;                  // copy the songlength
+
+    if(!AllocPositions(of.numpos)) return 0;
+    for(t=0; t<of.numpos; t++)
+        of.positions[t] = mh->positions[t];
+
+    // Count the number of patterns
+
+    of.numpat = 0;
+
+    for(t=0; t<of.numpos; t++)
+    {   if(of.positions[t] > of.numpat)
+            of.numpat = of.positions[t];
+    }
+    of.numpat++;
+    of.numtrk = of.numpat*of.numchn;
+    
+    // Finally, init the sampleinfo structures 
+    of.numins = of.numsmp = 31;
+
+    if(!AllocSamples())     return 0;
+    
+    s = mh->samples;       // init source pointer 
+    q = of.samples;
+    
+    for(t=0; t<of.numins; t++)
+    {   // convert the samplename
+        q->samplename = DupStr(s->samplename, 22);
+
+        // init the sampleinfo variables and
+        // convert the size pointers to longword format
+
+        q->speed     = finetune[s->finetune & 0xf];
+        q->volume    = s->volume;
+        q->loopstart = (ULONG)s->reppos << 1;
+        q->loopend   = q->loopstart + ((ULONG)s->replen << 1);
+        q->length    = (ULONG)s->length << 1;
+
+        q->flags     = SF_SIGNED;
+        if(s->replen > 1) q->flags |= SF_LOOP;
+
+        // fix replen if repend > length
+        if(q->loopend > q->length) q->loopend = q->length;
+
+        s++;    // point to next source sampleinfo
+        q++;
+    }
+
+    if(!ML_LoadPatterns()) return 0;
+    return 1;
+}
+
+
+CHAR *MOD_LoadTitle(void)
+{
+   CHAR s[20];
+
+   _mm_fseek(modfp,0,SEEK_SET);
+   if(!fread(s,20,1,modfp)) return NULL;
+   
+   return(DupStr(s,20));
+}
+
+
+MLOADER load_mod =
+{   NULL,
+    "Standard module",
+    "Portable MOD loader v0.11",
+    MOD_Init,
+    MOD_Test,
+    MOD_Load,
+    MOD_Cleanup,
+    MOD_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_mtm.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,282 @@
+/*
+
+ Name:  LOAD_MTM.C
+
+ Description:
+ MTM module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/**************************************************************************
+**************************************************************************/
+
+
+typedef struct MTMSAMPLE
+{   CHAR  samplename[22];
+    ULONG length;
+    ULONG reppos;
+    ULONG repend;
+    UBYTE finetune;
+    UBYTE volume;
+    UBYTE attribute;
+} MTMSAMPLE;
+
+
+
+typedef struct MTMHEADER
+{   UBYTE id[3];                    // MTM file marker
+    UBYTE version;                  // upper major, lower nibble minor version number
+    char  songname[20];             // ASCIIZ songname
+    UWORD numtracks;                // number of tracks saved
+    UBYTE lastpattern;              // last pattern number saved
+    UBYTE lastorder;                // last order number to play (songlength-1)
+    UWORD commentsize;              // length of comment field
+    UBYTE numsamples;               // number of samples saved 
+    UBYTE attribute;                // attribute byte (unused)
+    UBYTE beatspertrack;            //
+    UBYTE numchannels;              // number of channels used 
+    UBYTE panpos[32];               // voice pan positions
+} MTMHEADER;
+
+
+typedef struct MTMNOTE
+{   UBYTE a,b,c;
+} MTMNOTE;
+
+
+/**************************************************************************
+**************************************************************************/
+
+
+static MTMHEADER *mh = NULL;
+static MTMNOTE *mtmtrk = NULL;
+static UWORD pat[32];
+
+char MTM_Version[] = "MTM";
+
+
+
+BOOL MTM_Test(void)
+{
+    UBYTE id[3];
+    if(!_mm_read_UBYTES(id,3,modfp)) return 0;
+    if(!memcmp(id,"MTM",3)) return 1;
+    return 0;
+}
+
+
+BOOL MTM_Init(void)
+{
+    if(!(mtmtrk=(MTMNOTE *)_mm_calloc(64,sizeof(MTMNOTE)))) return 0;
+    if(!(mh=(MTMHEADER *)_mm_calloc(1,sizeof(MTMHEADER)))) return 0;
+
+    return 1;
+}
+
+
+void MTM_Cleanup(void)
+{
+    if(mtmtrk!=NULL) free(mtmtrk);
+    if(mh!=NULL) free(mh);
+
+    mtmtrk = NULL;
+    mh = NULL;
+}
+
+
+UBYTE *MTM_Convert(void)
+{
+    int t;
+    UBYTE a,b,c,inst,note,eff,dat;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {   a = mtmtrk[t].a;
+        b = mtmtrk[t].b;
+        c = mtmtrk[t].c;
+
+        inst = ((a&0x3)<<4)|(b>>4);
+        note = a>>2;
+
+        eff = b&0xf;
+        dat = c;
+
+        if(inst!=0)  UniInstrument(inst-1);
+        if(note!=0)  UniNote(note+24);
+
+        // mtm bug bugfix: when the effect is volslide,
+        // slide-up _always_ overrides slide-dn.
+
+        if(eff==0xa && dat&0xf0) dat&=0xf0;
+
+        // Convert pattern jump from Dec to Hex
+        if(eff == 0xd)
+            dat = (((dat&0xf0)>>4)*10)+(dat&0xf);
+        UniPTEffect(eff,dat);
+        UniNewline();
+    }
+    return UniDup();
+}
+
+
+BOOL MTM_Load(void)
+{
+    MTMSAMPLE s;
+    SAMPLE *q;
+
+    int t,u;
+
+    // try to read module header 
+
+    _mm_read_UBYTES(mh->id,3,modfp);
+    mh->version     =_mm_read_UBYTE(modfp);
+    _mm_read_string(mh->songname,20,modfp);
+    mh->numtracks   =_mm_read_I_UWORD(modfp);
+    mh->lastpattern =_mm_read_UBYTE(modfp);
+    mh->lastorder   =_mm_read_UBYTE(modfp);
+    mh->commentsize =_mm_read_I_UWORD(modfp);
+    mh->numsamples  =_mm_read_UBYTE(modfp);
+    mh->attribute   =_mm_read_UBYTE(modfp);
+    mh->beatspertrack=_mm_read_UBYTE(modfp);
+    mh->numchannels =_mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(mh->panpos,32,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+
+    of.initspeed = 6;
+    of.inittempo = 125;
+    of.modtype   = strdup(MTM_Version);
+    of.numchn    = mh->numchannels;
+    of.numtrk    = mh->numtracks+1;         // get number of channels
+    of.songname  = DupStr(mh->songname,20); // make a cstr of songname
+    of.numpos    = mh->lastorder+1;         // copy the songlength
+    of.numpat    = mh->lastpattern+1;
+    for(t=0; t<32; t++) of.panning[t] = mh->panpos[t] << 4;
+
+    of.numins = of.numsmp = mh->numsamples;
+    if(!AllocSamples()) return 0;
+    
+    q = of.samples;
+    
+    for(t=0; t<of.numins; t++)
+    {   // try to read sample info
+        _mm_read_string(s.samplename,22,modfp);
+        s.length    =_mm_read_I_ULONG(modfp);
+        s.reppos    =_mm_read_I_ULONG(modfp);
+        s.repend    =_mm_read_I_ULONG(modfp);
+        s.finetune  =_mm_read_UBYTE(modfp);
+        s.volume    =_mm_read_UBYTE(modfp);
+        s.attribute =_mm_read_UBYTE(modfp);
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_SAMPLEINFO; 
+            return 0;
+        }
+
+        q->samplename = DupStr(s.samplename,22);
+        q->seekpos   = 0;
+        q->speed     = finetune[s.finetune];
+        q->length    = s.length;
+        q->loopstart = s.reppos;
+        q->loopend   = s.repend;
+        q->volume    = s.volume;
+
+        if((s.repend-s.reppos) > 2) q->flags |= SF_LOOP;
+
+        if(s.attribute & 1)
+        {   // If the sample is 16-bits, convert the length
+            // and replen byte-values into sample-values
+
+            q->flags|=SF_16BITS;
+            q->length>>=1;
+            q->loopstart>>=1;
+            q->loopend>>=1;
+        }
+
+        q++;
+    }
+
+    if(!AllocPositions(of.numpos)) return 0;
+    for(t=0; t<of.numpos; t++)
+        of.positions[t] = _mm_read_UBYTE(modfp);
+    for(; t<128; t++)  _mm_read_UBYTE(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    if(!AllocTracks()) return 0;
+    if(!AllocPatterns()) return 0;
+
+    of.tracks[0] = MTM_Convert();             // track 0 is empty
+
+    for(t=1; t<of.numtrk; t++)
+    {   int s;
+
+        for(s=0; s<64; s++)
+        {   mtmtrk[s].a=_mm_read_UBYTE(modfp);
+            mtmtrk[s].b=_mm_read_UBYTE(modfp);
+            mtmtrk[s].c=_mm_read_UBYTE(modfp);
+        }
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_TRACK;
+            return 0;
+        }
+
+        if(!(of.tracks[t]=MTM_Convert())) return 0;
+    }
+
+    for(t=0; t<of.numpat; t++)
+    {   _mm_read_I_UWORDS(pat,32,modfp);
+
+        for(u=0; u<of.numchn; u++)
+            of.patterns[((long)t*of.numchn)+u]=pat[u];
+    }
+
+    // read comment field
+
+    if(!ReadComment(mh->commentsize)) return 0;
+
+    return 1;
+}
+
+
+CHAR *MTM_LoadTitle(void)
+{
+    CHAR s[20];
+
+    _mm_fseek(modfp,4,SEEK_SET);
+    if(!fread(s,20,1,modfp)) return NULL;
+ 
+    return(DupStr(s,20));
+}
+                
+
+MLOADER load_mtm =
+{   NULL,
+    "MTM",
+    "Portable MTM loader v0.1",
+    MTM_Init,
+    MTM_Test,
+    MTM_Load,
+    MTM_Cleanup,
+    MTM_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_s3m.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,466 @@
+/*
+
+ Name: LOAD_S3M.C
+
+ Description:
+ Screamtracker (S3M) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/**************************************************************************
+**************************************************************************/
+
+typedef struct S3MNOTE
+{   UBYTE note,ins,vol,cmd,inf;
+} S3MNOTE;
+
+typedef S3MNOTE S3MTRACK[64];
+
+
+// Raw S3M header struct:
+
+typedef struct S3MHEADER
+{   CHAR  songname[28];
+    UBYTE t1a;
+    UBYTE type;
+    UBYTE unused1[2];
+    UWORD ordnum;
+    UWORD insnum;
+    UWORD patnum;
+    UWORD flags;
+    UWORD tracker;
+    UWORD fileformat;
+    CHAR  scrm[4];
+    UBYTE mastervol;
+    UBYTE initspeed;
+    UBYTE inittempo;
+    UBYTE mastermult;
+    UBYTE ultraclick;
+    UBYTE pantable;
+    UBYTE unused2[8];
+    UWORD special;
+    UBYTE channels[32];
+} S3MHEADER;
+
+
+// Raw S3M sampleinfo struct:
+
+typedef struct S3MSAMPLE
+{   UBYTE type;
+    CHAR  filename[12];
+    UBYTE memsegh;
+    UWORD memsegl;
+    ULONG length;
+    ULONG loopbeg;
+    ULONG loopend;
+    UBYTE volume;
+    UBYTE dsk;
+    UBYTE pack;
+    UBYTE flags;
+    ULONG c2spd;
+    UBYTE unused[12];
+    CHAR  sampname[28];
+    CHAR  scrs[4];
+} S3MSAMPLE;
+
+/**************************************************************************
+**************************************************************************/
+
+
+extern UBYTE *poslookup;       // S3M/IT fix - removing blank patterns needs a
+                               // lookup table to fix position-jump commands
+extern SBYTE  remap[64];       // for removing empty channels
+
+static S3MNOTE   *s3mbuf  = NULL; // pointer to a complete S3M pattern
+static S3MHEADER *mh      = NULL;
+static UWORD     *paraptr = NULL; // parapointer array (see S3M docs)
+
+CHAR  S3M_Version[] = "Screamtracker 3.xx";
+
+BOOL S3M_Test(void)
+{
+    UBYTE id[4];
+    
+    _mm_fseek(modfp,0x2c,SEEK_SET);
+    if(!_mm_read_UBYTES(id,4,modfp)) return 0;
+    if(!memcmp(id,"SCRM",4)) return 1;
+    return 0;
+}
+
+BOOL S3M_Init(void)
+{
+    if(!(s3mbuf    = (S3MNOTE *)_mm_malloc(16*64*sizeof(S3MNOTE)))) return 0;
+    if(!(mh        = (S3MHEADER *)_mm_calloc(1,sizeof(S3MHEADER)))) return 0;
+    if(!(poslookup = (UBYTE *)_mm_malloc(sizeof(UBYTE)*128))) return 0;
+
+    return 1;
+}
+
+void S3M_Cleanup(void)
+{
+    if(s3mbuf!=NULL) free(s3mbuf);
+    if(paraptr!=NULL) free(paraptr);
+    if(poslookup!=NULL) free(poslookup);
+    if(mh!=NULL) free(mh);
+
+    paraptr   = NULL;
+    s3mbuf    = NULL;
+    poslookup = NULL;
+    mh        = NULL;
+}
+
+
+BOOL S3M_GetNumChannels(void)
+
+// Because so many s3m files have 16 channels as the set number used, but really
+// only use far less (usually 8 to 12 still), I had to make this function,
+// which determines the number of channels that are actually USED by a pattern.
+//
+// For every channel that's used, it sets the appropriate array entry of the
+// global varialbe 'isused'
+//
+// NOTE: You must first seek to the file location of the pattern before calling
+//       this procedure.
+// Returns 1 on fail.
+
+{
+    int row=0,flag,ch;
+
+    while(row<64)
+    {   flag = _mm_read_UBYTE(modfp);
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 1;
+        }
+
+        if(flag)
+        {   ch = flag&31;
+            if(mh->channels[ch] < 16) remap[ch] = 0;
+            
+            if(flag&32)
+            {   _mm_read_UBYTE(modfp);
+                _mm_read_UBYTE(modfp);
+            }
+
+            if(flag&64)
+                _mm_read_UBYTE(modfp);
+
+            if(flag&128)
+            {   _mm_read_UBYTE(modfp);
+                _mm_read_UBYTE(modfp);
+            }
+        } else row++;
+    }
+
+    return 0;
+}    
+
+
+BOOL S3M_ReadPattern(void)
+{
+    int row=0,flag,ch;
+    S3MNOTE *n;
+    S3MNOTE dummy;
+
+    // clear pattern data
+    memset(s3mbuf,255,16*64*sizeof(S3MNOTE));
+
+    while(row<64)
+    {   flag = _mm_read_UBYTE(modfp);
+
+        if(flag==EOF)
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 0;
+        }
+
+        if(flag)
+        {   ch = remap[flag&31];
+
+            if(ch != -1)
+                n = &s3mbuf[(64U*ch)+row];
+            else
+                n = &dummy;
+
+            if(flag&32)
+            {   n->note = _mm_read_UBYTE(modfp);
+                n->ins  = _mm_read_UBYTE(modfp);
+            }
+
+            if(flag&64)
+                n->vol = _mm_read_UBYTE(modfp);
+
+            if(flag&128)
+            {   n->cmd = _mm_read_UBYTE(modfp);
+                n->inf = _mm_read_UBYTE(modfp);
+            }
+        } else row++;
+    }
+    return 1;
+}
+
+
+void S3MIT_ProcessCmd(UBYTE cmd, UBYTE inf, BOOL oldeffect);
+
+UBYTE *S3M_ConvertTrack(S3MNOTE *tr)
+{
+    int t;
+
+    UBYTE note,ins,vol;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {
+        note = tr[t].note;
+        ins  = tr[t].ins;
+        vol  = tr[t].vol;
+
+
+        if(ins!=0 && ins!=255) UniInstrument(ins-1);
+        if(note!=255)
+        {   if(note==254) UniPTEffect(0xc,0);             // <- note off command
+            else UniNote(((note>>4)*12)+(note&0xf));      // <- normal note
+        }
+
+        if(vol<255)
+            UniPTEffect(0xc,vol);
+
+        S3MIT_ProcessCmd(tr[t].cmd,tr[t].inf,1);
+        UniNewline();
+    }
+
+    return UniDup();
+}
+
+
+BOOL S3M_Load(void)
+{
+    int    t,u,track = 0;
+    SAMPLE *q;
+    UBYTE  pan[32];
+
+    // try to read module header
+
+    _mm_read_string(mh->songname,28,modfp);
+    mh->t1a         =_mm_read_UBYTE(modfp);
+    mh->type        =_mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(mh->unused1,2,modfp);
+    mh->ordnum      =_mm_read_I_UWORD(modfp);
+    mh->insnum      =_mm_read_I_UWORD(modfp);
+    mh->patnum      =_mm_read_I_UWORD(modfp);
+    mh->flags       =_mm_read_I_UWORD(modfp);
+    mh->tracker     =_mm_read_I_UWORD(modfp);
+    mh->fileformat  =_mm_read_I_UWORD(modfp);
+    _mm_read_string(mh->scrm,4,modfp);
+
+    mh->mastervol   =_mm_read_UBYTE(modfp);
+    mh->initspeed   =_mm_read_UBYTE(modfp);
+    mh->inittempo   =_mm_read_UBYTE(modfp);
+    mh->mastermult  =_mm_read_UBYTE(modfp);
+    mh->ultraclick  =_mm_read_UBYTE(modfp);
+    mh->pantable    =_mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(mh->unused2,8,modfp);
+    mh->special     =_mm_read_I_UWORD(modfp);
+    _mm_read_UBYTES(mh->channels,32,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+
+    of.modtype     = strdup(S3M_Version);
+    of.modtype[14] = ((mh->tracker >> 8) &0xf) + 0x30;
+    of.modtype[16] = ((mh->tracker >> 4)&0xf) + 0x30;
+    of.modtype[17] = ((mh->tracker)&0xf) + 0x30;
+    of.songname    = DupStr(mh->songname,28);
+    of.numpat      = mh->patnum;
+    of.reppos      = 0;
+    of.numins      = of.numsmp = mh->insnum;
+    of.initspeed   = mh->initspeed;
+    of.inittempo   = mh->inittempo;
+    of.initvolume  = mh->mastervol<<1;
+
+    // read the order data
+    if(!AllocPositions(mh->ordnum)) return 0;
+    for(t=0; t<mh->ordnum; t++)
+        of.positions[t] = _mm_read_UBYTE(modfp);
+
+    of.numpos = 0;
+    for(t=0; t<mh->ordnum; t++)
+    {   of.positions[of.numpos] = of.positions[t];
+        poslookup[t]            = of.numpos;   // bug fix for FREAKY S3Ms
+        if(of.positions[t]<254) of.numpos++;        
+    }
+
+    if((paraptr=(UWORD *)_mm_malloc((of.numins+of.numpat)*sizeof(UWORD)))==NULL) return 0;
+
+    // read the instrument+pattern parapointers
+    _mm_read_I_UWORDS(paraptr,of.numins+of.numpat,modfp);
+
+
+    if(mh->pantable==252)
+    {   // read the panning table (ST 3.2 addition.  See below for further
+        // portions of channel panning [past reampper]).
+        _mm_read_UBYTES(pan,32,modfp);
+    }
+
+
+    // now is a good time to check if the header was too short :)
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+
+    // ==============================================
+    // Load those darned Samples!  (no insts in ST3)
+
+    if(!AllocSamples()) return 0;
+
+    q = of.samples;
+
+    for(t=0; t<of.numins; t++)
+    {   S3MSAMPLE s;
+
+        // seek to instrument position
+
+        _mm_fseek(modfp,((long)paraptr[t])<<4,SEEK_SET);
+
+        // and load sample info
+
+        s.type      =_mm_read_UBYTE(modfp);
+        _mm_read_string(s.filename,12,modfp);
+        s.memsegh   =_mm_read_UBYTE(modfp);
+        s.memsegl   =_mm_read_I_UWORD(modfp);
+        s.length    =_mm_read_I_ULONG(modfp);
+        s.loopbeg   =_mm_read_I_ULONG(modfp);
+        s.loopend   =_mm_read_I_ULONG(modfp);
+        s.volume    =_mm_read_UBYTE(modfp);
+        s.dsk       =_mm_read_UBYTE(modfp);
+        s.pack      =_mm_read_UBYTE(modfp);
+        s.flags     =_mm_read_UBYTE(modfp);
+        s.c2spd     =_mm_read_I_ULONG(modfp);
+        _mm_read_UBYTES(s.unused,12,modfp);
+        _mm_read_string(s.sampname,28,modfp);
+        _mm_read_string(s.scrs,4,modfp);
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+            return 0;
+        }
+
+        q->samplename = DupStr(s.sampname,28);
+        q->speed      = s.c2spd;
+        q->length     = s.length;
+        q->loopstart  = s.loopbeg;
+        q->loopend    = s.loopend;
+        q->volume     = s.volume;
+        q->seekpos    = (((long)s.memsegh)<<16|s.memsegl)<<4;
+
+        if(s.flags&1) q->flags |= SF_LOOP;
+        if(s.flags&4) q->flags |= SF_16BITS;
+        if(mh->fileformat==1) q->flags |= SF_SIGNED;
+
+        // DON'T load sample if it doesn't have the SCRS tag
+        if(memcmp(s.scrs,"SCRS",4)!=0) q->length = 0;
+
+        q++;
+    }
+
+    // ====================================
+    // Determine the number of channels actually used.  (what ever happened
+    // to the concept of a single "numchn" variable, eh?!
+
+    of.numchn = 0;
+    memset(remap,-1,32*sizeof(UBYTE));
+
+    for(t=0; t<of.numpat; t++)
+    {   // seek to pattern position ( + 2 skip pattern length )
+        _mm_fseek(modfp,(long)((paraptr[of.numins+t])<<4)+2,SEEK_SET);
+        if(S3M_GetNumChannels()) return 0;
+    }
+    
+    // build the remap array 
+    for(t=0; t<32; t++)
+    {   if(remap[t]==0)
+        {   remap[t] = of.numchn;
+            of.numchn++;
+        }
+    }
+
+    // ============================================================
+    // set panning positions AFTER building remap chart!
+
+    for(t=0; t<32; t++)
+    {   if((mh->channels[t]<16) && (remap[t]!=-1))
+        {   if(mh->channels[t]<8)
+                of.panning[remap[t]] = 0x20;     // 0x30 = std s3m val
+            else
+                of.panning[remap[t]] = 0xd0;     // 0xc0 = std s3m val
+        }
+    }
+
+    if(mh->pantable==252)
+    {   // set panning positions according to panning table (new for st3.2)
+        for(t=0; t<32; t++)
+        {   if((pan[t]&0x20) && (mh->channels[t]<16) && (remap[t]!=-1))
+                of.panning[remap[t]] = (pan[t]&0xf)<<4;
+        }
+    }
+
+
+    // ==============================
+    // Load the pattern info now!
+    
+    of.numtrk = of.numpat*of.numchn;
+    if(!AllocTracks()) return 0;
+    if(!AllocPatterns()) return 0;
+
+    for(t=0; t<of.numpat; t++)
+    {   // seek to pattern position ( + 2 skip pattern length )
+        _mm_fseek(modfp,(((long)paraptr[of.numins+t])<<4)+2,SEEK_SET);
+        if(!S3M_ReadPattern()) return 0;
+        for(u=0; u<of.numchn; u++)
+            if(!(of.tracks[track++]=S3M_ConvertTrack(&s3mbuf[u*64]))) return 0;
+    }
+
+    return 1;
+}
+
+         
+CHAR *S3M_LoadTitle(void)
+{
+   CHAR s[28];
+
+   _mm_fseek(modfp,0,SEEK_SET);
+   if(!fread(s,28,1,modfp)) return NULL;
+   
+   return(DupStr(s,28));
+}
+
+
+MLOADER load_s3m =
+{   NULL,
+    "S3M",
+    "S3M loader v0.3",
+    S3M_Init,
+    S3M_Test,
+    S3M_Load,
+    S3M_Cleanup,
+
+    S3M_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_stm.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,374 @@
+/*
+
+ Name: LOAD_STM.C
+
+ Description:
+ ScreamTracker 2 (STM) module Loader - Version 1.oOo Release 2 
+ A Coding Nightmare by Rao and Air Richter of HaRDCoDE
+ You can now play all of those wonderful old C.C. Catch STM's!
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author). 
+
+*/
+
+#include <string.h>
+#include <ctype.h>
+#include "mikmod.h"
+
+
+typedef struct STMNOTE
+{   UBYTE note,insvol,volcmd,cmdinf;
+} STMNOTE;
+
+
+// Raw STM sampleinfo struct:
+
+typedef struct STMSAMPLE
+{  CHAR  filename[12];   // Can't have long comments - just filename comments :)
+   UBYTE unused;         // 0x00
+   UBYTE instdisk;       // Instrument disk
+   UWORD reserved;       // ISA in memory when in ST 2
+   UWORD length;         // Sample length
+   UWORD loopbeg;        // Loop start point
+   UWORD loopend;        // Loop end point
+   UBYTE volume;         // Volume
+   UBYTE reserved2;      // More reserved crap
+   UWORD c2spd;          // Good old c2spd
+   UBYTE reserved3[4];   // Yet more of PSi's reserved crap
+   UWORD isa;            // Internal Segment Address ->
+                         //    contrary to the tech specs, this is NOT actually
+                         //    written to the stm file.
+} STMSAMPLE;
+
+// Raw STM header struct:
+
+typedef struct STMHEADER
+{  CHAR  songname[20];
+   CHAR  trackername[8];   // !SCREAM! for ST 2.xx 
+   UBYTE unused;           // 0x1A 
+   UBYTE filetype;         // 1=song, 2=module (only 2 is supported, of course) :) 
+   UBYTE ver_major;        // Like 2 
+   UBYTE ver_minor;        // "ditto" 
+   UBYTE inittempo;        // initspeed= stm inittempo>>4 
+   UBYTE  numpat;          // number of patterns 
+   UBYTE   globalvol;      // <- WoW! a RiGHT TRiANGLE =8*) 
+   UBYTE    reserved[13];  // More of PSi's internal crap 
+   STMSAMPLE sample[31];   // STM sample data
+   UBYTE patorder[128];    // Docs say 64 - actually 128
+} STMHEADER;
+
+
+static STMNOTE *stmbuf = NULL;
+static STMHEADER *mh = NULL;
+
+static CHAR  STM_Version[] = "Screamtracker 2";
+
+
+BOOL STM_Test(void)
+{
+   UBYTE str[9],filetype;
+
+   _mm_fseek(modfp,21,SEEK_SET);
+   _mm_read_UBYTES(str,9,modfp);
+   filetype = _mm_read_UBYTE(modfp);
+   if(!memcmp(str,"!SCREAM!",8) || (filetype!=2)) // STM Module = filetype 2
+      return 0;
+   return 1;
+}
+
+
+
+BOOL STM_Init(void)
+{
+    if(!(mh=(STMHEADER *)_mm_calloc(1,sizeof(STMHEADER)))) return 0;
+    if(!(stmbuf=(STMNOTE *)_mm_calloc(64U*of.numchn,sizeof(STMNOTE)))) return 0;
+
+    return 1;
+}
+
+void STM_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    if(stmbuf!=NULL) free(stmbuf);
+
+    stmbuf = NULL;
+    mh = NULL;
+}
+
+
+
+void STM_ConvertNote(STMNOTE *n)
+{
+    UBYTE note,ins,vol,cmd,inf;
+
+    // extract the various information from the 4 bytes that
+    //  make up a single note
+
+    note = n->note;
+    ins  = n->insvol>>3;
+    vol  = (n->insvol&7)+(n->volcmd>>1);
+    cmd  = n->volcmd&15;
+    inf  = n->cmdinf;
+
+    if(ins!=0 && ins<32)  UniInstrument(ins-1);
+
+  // special values of [SBYTE0] are handled here ->
+  // we have no idea if these strange values will ever be encountered.
+  // but it appears as though stms sound correct.
+  if(note==254 || note==252) UniPTEffect(0xc,0); // <- note off command (???)
+     else
+  // if note < 251, then all three bytes are stored in the file
+  if(note<251) UniNote((((note>>4)+2)*12)+(note&0xf));      // <- normal note and up the octave by two
+
+    if(vol<65)
+        UniPTEffect(0xc,vol);
+
+    if(cmd!=255){
+        switch(cmd){
+
+            case 1:                 // Axx set speed to xx and add 0x1c to fix StoOoPiD STM 2.x
+                UniPTEffect(0xf,inf>>4);
+                break;
+
+            case 2:                 // Bxx position jump
+                UniPTEffect(0xb,inf);
+                break;
+
+            case 3:                 // Cxx patternbreak to row xx
+                UniPTEffect(0xd,(((inf&0xf0)>>4)*10)+(inf&0xf));
+                break;
+
+            case 4:                 // Dxy volumeslide
+                UniWrite(UNI_S3MEFFECTD);
+                UniWrite(inf);
+                break;
+
+            case 5:                 // Exy toneslide down
+                UniWrite(UNI_S3MEFFECTE);
+                UniWrite(inf);
+                break;
+
+            case 6:                 // Fxy toneslide up
+                UniWrite(UNI_S3MEFFECTF);
+                UniWrite(inf);
+                break;
+
+            case 7:                 // Gxx Tone portamento,speed xx
+                UniPTEffect(0x3,inf);
+                break;
+
+            case 8:                 // Hxy vibrato
+                UniPTEffect(0x4,inf);
+                break;
+
+            case 9:                 // Ixy tremor, ontime x, offtime y
+                UniWrite(UNI_S3MEFFECTI);
+                UniWrite(inf);
+                break;
+
+            case 0xa:               // Jxy arpeggio
+                UniPTEffect(0x0,inf);
+                break;
+
+            case 0xb:               // Kxy Dual command H00 & Dxy
+                UniPTEffect(0x4,0);
+                UniWrite(UNI_S3MEFFECTD);
+                UniWrite(inf);
+                break;
+
+            case 0xc:               // Lxy Dual command G00 & Dxy
+                UniPTEffect(0x3,0);
+                UniWrite(UNI_S3MEFFECTD);
+                UniWrite(inf);
+                break;
+
+         // Support all these above, since ST2 can LOAD these values
+         // but can actually only play up to J - and J is only
+         // half-way implemented in ST2
+
+            case 0x18:      // Xxx amiga command 8xx - What the hell, support panning. :)
+                UniPTEffect(0x8,inf);
+                break;
+        }
+    }
+}
+
+
+UBYTE *STM_ConvertTrack(STMNOTE *n)
+{
+    int t;
+
+    UniReset();
+    for(t=0; t<64; t++)
+    {   STM_ConvertNote(n);
+        UniNewline();
+        n+=of.numchn;
+    }
+    return UniDup();
+}
+
+
+BOOL STM_LoadPatterns(void)
+{
+    int t,s,tracks=0;
+
+    if(!AllocPatterns()) return 0;
+    if(!AllocTracks()) return 0;
+
+    // Allocate temporary buffer for loading
+    // and converting the patterns
+
+    for(t=0; t<of.numpat; t++)
+    {   for(s=0; s<(64U*of.numchn); s++)
+        {   stmbuf[s].note   = _mm_read_UBYTE(modfp);
+            stmbuf[s].insvol = _mm_read_UBYTE(modfp);
+            stmbuf[s].volcmd = _mm_read_UBYTE(modfp);
+            stmbuf[s].cmdinf = _mm_read_UBYTE(modfp);
+        }
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 0;
+        }
+
+        for(s=0;s<of.numchn;s++)
+        {   if(!(of.tracks[tracks++]=STM_ConvertTrack(stmbuf+s))) return 0;
+        }
+    }
+
+    return 1;
+}
+
+
+
+BOOL STM_Load(void)
+{
+    int        t; 
+    ULONG      MikMod_ISA; // We MUST generate our own ISA - NOT stored in the stm
+    SAMPLE     *q;
+
+    // try to read stm header
+
+    _mm_read_string(mh->songname,20,modfp);
+    _mm_read_string(mh->trackername,8,modfp);
+    mh->unused      =_mm_read_UBYTE(modfp);
+    mh->filetype    =_mm_read_UBYTE(modfp);
+    mh->ver_major   =_mm_read_UBYTE(modfp);
+    mh->ver_minor   =_mm_read_UBYTE(modfp);
+    mh->inittempo   =_mm_read_UBYTE(modfp);
+    mh->numpat      =_mm_read_UBYTE(modfp);
+    mh->globalvol   =_mm_read_UBYTE(modfp);
+    _mm_read_UBYTES(mh->reserved,13,modfp);
+
+    for(t=0;t<31;t++)
+    {   STMSAMPLE *s = &mh->sample[t];  // STM sample data
+
+        _mm_read_string(s->filename,12,modfp);
+        s->unused   =_mm_read_UBYTE(modfp);
+        s->instdisk =_mm_read_UBYTE(modfp);
+        s->reserved =_mm_read_I_UWORD(modfp);
+        s->length   =_mm_read_I_UWORD(modfp);
+        s->loopbeg  =_mm_read_I_UWORD(modfp);
+        s->loopend  =_mm_read_I_UWORD(modfp);
+        s->volume   =_mm_read_UBYTE(modfp);
+        s->reserved2=_mm_read_UBYTE(modfp);
+        s->c2spd    =_mm_read_I_UWORD(modfp);
+        _mm_read_UBYTES(s->reserved3,4,modfp);
+        s->isa      =_mm_read_I_UWORD(modfp);
+    }
+    _mm_read_UBYTES(mh->patorder,128,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+
+    of.modtype   = strdup(STM_Version);
+    of.songname  = DupStr(mh->songname,20); // make a cstr of songname
+    of.numpat    = mh->numpat;
+    of.inittempo = 125;                     // mh->inittempo+0x1c;
+    of.initspeed = mh->inittempo>>4;
+    of.numchn    = 4;                       // get number of channels
+
+    t=0;
+    if(!AllocPositions(0x80)) return 0;
+    while(mh->patorder[t]!=99)    // 99 terminates the patorder list
+    {   of.positions[t] = mh->patorder[t];
+        t++;
+    }
+    of.numpos = --t;
+    of.numtrk = of.numpat*of.numchn;
+
+    // Finally, init the sampleinfo structures
+    of.numins = of.numsmp = 31;    // always this
+
+    if(!AllocSamples()) return 0;
+    if(!STM_LoadPatterns()) return 0;
+
+    q = of.samples;
+
+    MikMod_ISA = _mm_ftell(modfp);
+    MikMod_ISA = (MikMod_ISA+15)&0xfffffff0;
+
+    for(t=0; t<of.numins; t++)
+    {   // load sample info
+
+        q->samplename = DupStr(mh->sample[t].filename,12);
+        q->speed      = mh->sample[t].c2spd;
+        q->volume     = mh->sample[t].volume;
+        q->length     = mh->sample[t].length;
+        if (!mh->sample[t].volume || q->length==1 ) q->length = 0; // if vol = 0 or length = 1, then no sample
+        q->loopstart  = mh->sample[t].loopbeg;
+        q->loopend    = mh->sample[t].loopend;
+        q->seekpos    = MikMod_ISA;
+
+        MikMod_ISA+=q->length;
+        MikMod_ISA = (MikMod_ISA+15)&0xfffffff0;
+
+        // Once again, contrary to the STM specs, all the sample data is
+        // actually SIGNED! Sheesh
+
+        q->flags = SF_SIGNED;
+
+        if(mh->sample[t].loopend>0 && mh->sample[t].loopend!=0xffff) q->flags|=SF_LOOP;
+
+        // fix replen if repend>length
+
+        if(q->loopend > q->length) q->loopend = q->length;
+
+        q++;
+    }
+
+    return 1;
+}
+
+
+CHAR *STM_LoadTitle(void)
+{
+    CHAR s[20];
+
+    _mm_fseek(modfp,0,SEEK_SET);
+    if(!fread(s,20,1,modfp)) return NULL;
+
+    return(DupStr(s,20));
+}
+
+
+MLOADER load_stm =
+{   NULL,
+    "STM",
+    "Portable STM Loader - V 1.2",
+    STM_Init,
+    STM_Test,
+    STM_Load,
+    STM_Cleanup,
+    STM_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_ult.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,292 @@
+/*
+
+ Name:  LOAD_ULT.C
+
+ Description:
+ Ultratracker (ULT) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+
+#define ULTS_16BITS     4
+#define ULTS_LOOP       8
+#define ULTS_REVERSE    16
+
+
+// Raw ULT header struct:
+
+typedef struct ULTHEADER
+{   CHAR  id[16];
+    CHAR  songtitle[32];
+    UBYTE reserved;
+} ULTHEADER;
+
+
+// Raw ULT sampleinfo struct:
+
+typedef struct ULTSAMPLE
+{   CHAR   samplename[32];
+    CHAR   dosname[12];
+    SLONG  loopstart;
+    SLONG  loopend;
+    SLONG  sizestart;
+    SLONG  sizeend;
+    UBYTE  volume;
+    UBYTE  flags;
+    SWORD  finetune;
+} ULTSAMPLE;
+
+
+typedef struct ULTEVENT
+{   UBYTE note,sample,eff,dat1,dat2;
+} ULTEVENT;
+
+
+CHAR *ULT_Version[]=
+{   "Ultra Tracker V1.3",
+    "Ultra Tracker V1.4",
+    "Ultra Tracker V1.5",
+    "Ultra Tracker V1.6"
+};
+
+
+BOOL ULT_Test(void)
+{
+    CHAR id[16];
+
+    if(!_mm_read_string(id,15,modfp)) return 0;
+    return(!strncmp(id,"MAS_UTrack_V00",14));
+}
+
+
+BOOL ULT_Init(void)
+{
+    return 1;
+}
+
+
+void ULT_Cleanup(void)
+{
+}
+
+ULTEVENT ev;
+
+
+
+int ReadUltEvent(ULTEVENT *event)
+{
+    UBYTE flag,rep=1;
+
+    flag = _mm_read_UBYTE(modfp);
+
+    if(flag==0xfc)
+    {   rep = _mm_read_UBYTE(modfp);
+        event->note =_mm_read_UBYTE(modfp);
+    } else
+        event->note = flag;
+
+    event->sample   =_mm_read_UBYTE(modfp);
+    event->eff      =_mm_read_UBYTE(modfp);
+    event->dat1     =_mm_read_UBYTE(modfp);
+    event->dat2     =_mm_read_UBYTE(modfp);
+
+    return rep;
+}
+
+
+BOOL ULT_Load(void)
+{
+    int        t,u,tracks=0;
+    SAMPLE     *q;
+    ULTSAMPLE  s;
+    ULTHEADER  mh;
+    UBYTE      nos,noc,nop;
+
+    // try to read module header
+
+    _mm_read_string(mh.id,15,modfp);
+    _mm_read_string(mh.songtitle,32,modfp);
+    mh.reserved = _mm_read_UBYTE(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    if(mh.id[14]<'1' || mh.id[14]>'4')
+    {   _mm_errno = MMERR_NOT_A_MODULE;
+        return 0;
+    }
+
+    of.modtype   = strdup(ULT_Version[mh.id[14]-'1']);
+    of.initspeed = 6;
+    of.inittempo = 125;
+
+    // read songtext
+
+    if(!ReadComment((UWORD)mh.reserved*32)) return 0;
+
+    nos = _mm_read_UBYTE(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    of.songname = DupStr(mh.songtitle,32);
+    of.numins   = nos;
+
+    if(!AllocSamples()) return 0;
+    
+    q = of.samples;
+
+    for(t=0; t<nos; t++)
+    {   // try to read sample info
+
+        _mm_read_string(s.samplename,32,modfp);
+        _mm_read_string(s.dosname,12,modfp);
+        s.loopstart     =_mm_read_I_ULONG(modfp);
+        s.loopend       =_mm_read_I_ULONG(modfp);
+        s.sizestart     =_mm_read_I_ULONG(modfp);
+        s.sizeend       =_mm_read_I_ULONG(modfp);
+        s.volume        =_mm_read_UBYTE(modfp);
+        s.flags         =_mm_read_UBYTE(modfp);
+        s.finetune      =_mm_read_I_SWORD(modfp);
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+            return 0;
+        }
+
+        q->samplename = DupStr(s.samplename,32);
+        q->speed      = 8363;
+
+        if(mh.id[14]>='4')
+        {   _mm_read_I_UWORD(modfp);       // read 1.6 extra info(??) word
+            q->speed=s.finetune;
+        }
+
+        q->length    = s.sizeend-s.sizestart;
+        q->volume    = s.volume>>2;
+        q->loopstart = s.loopstart;
+        q->loopend   = s.loopend;
+
+        q->flags = SF_SIGNED;
+
+        if(s.flags&ULTS_LOOP)
+            q->flags|=SF_LOOP;
+
+        if(s.flags&ULTS_16BITS)
+        {   q->flags|=SF_16BITS;
+            q->loopstart>>=1;
+            q->loopend>>=1;
+        }
+
+        q++;
+    }
+
+    if(!AllocPositions(256)) return 0;
+    for(t=0; t<256; t++)
+        of.positions[t] = _mm_read_UBYTE(modfp);
+    for(t=0; t<256; t++)
+        if(of.positions[t]==255) break;
+
+    of.numpos = t;
+
+    noc = _mm_read_UBYTE(modfp);
+    nop = _mm_read_UBYTE(modfp);
+
+    of.numchn = noc+1;
+    of.numpat = nop+1;
+    of.numtrk = of.numchn*of.numpat;
+
+    if(!AllocTracks()) return 0;
+    if(!AllocPatterns()) return 0;
+
+    for(u=0; u<of.numchn; u++)
+    {   for(t=0; t<of.numpat; t++)
+            of.patterns[(t*of.numchn)+u]=tracks++;
+    }
+
+    // read pan position table for v1.5 and higher
+
+    if(mh.id[14]>='3')
+        for(t=0; t<of.numchn; t++) of.panning[t]=_mm_read_UBYTE(modfp)<<4;
+
+    for(t=0; t<of.numtrk; t++)
+    {   int rep,s,done;
+
+        UniReset();
+        done=0;
+
+        while(done<64)
+        {   rep=ReadUltEvent(&ev);
+
+            if(feof(modfp))
+            {   _mm_errno = MMERR_LOADING_TRACK;
+                return 0;
+            }
+
+            for(s=0; s<rep; s++)
+            {   UBYTE eff;
+
+                if(ev.sample) UniInstrument(ev.sample-1);
+                if(ev.note)   UniNote(ev.note+23);
+
+                eff = ev.eff>>4;
+
+                //  ULT panning effect fixed by Alexander Kerkhove :
+
+                if(eff==0xc) UniPTEffect(eff,ev.dat2>>2);
+                else if(eff==0xb) UniPTEffect(8,ev.dat2*0xf);
+                else UniPTEffect(eff,ev.dat2);
+
+                eff=ev.eff&0xf;
+
+                if(eff==0xc) UniPTEffect(eff,ev.dat1>>2);
+                else if(eff==0xb) UniPTEffect(8,ev.dat1*0xf);
+                else UniPTEffect(eff,ev.dat1);
+
+                UniNewline();
+                done++;
+            }
+        }
+        if(!(of.tracks[t]=UniDup())) return 0;
+    }
+
+    return 1;
+}
+
+
+CHAR *ULT_LoadTitle(void)
+{
+   CHAR s[32];
+
+   _mm_fseek(modfp,15,SEEK_SET);
+   if(!fread(s,32,1,modfp)) return NULL;
+   
+   return(DupStr(s,32));
+}
+
+
+MLOADER load_ult =
+{   NULL,
+    "ULT",
+    "Portable ULT loader v0.1",
+    ULT_Init,
+    ULT_Test,
+    ULT_Load,
+    ULT_Cleanup,
+    ULT_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_uni.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,229 @@
+/*
+
+Name: LOAD_UNI.C
+
+Description:
+UNIMOD (mikmod's internal format) module loader.
+Currently only supports UniMOD 06 - the internal format for MikMod 3.0.
+
+Portability:
+All systems - all compilers (hopefully)
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+
+BOOL UNI_Test(void)
+{
+    UBYTE id[4];
+
+    _mm_read_UBYTES(id,4,modfp);
+    if(!memcmp(id, "UN06", 4)) return 1;
+
+    return 0;
+}
+
+
+BOOL UNI_Init(void)
+{
+    return 1;
+}
+
+
+void UNI_Cleanup(void)
+{
+}
+
+
+UBYTE *TrkRead(void)
+{
+    UBYTE  *t;
+    UWORD  len;
+
+    len = _mm_read_M_UWORD(modfp);
+    t   = (UBYTE *)malloc(len);
+    _mm_read_UBYTES(t,len,modfp);
+
+    return t;
+}
+
+
+BOOL UNI_Load(void)
+{
+    int t,v,w;
+    INSTRUMENT *i;
+    SAMPLE *s;
+
+    // UNI format version 3.0 (#6)
+    
+    of.modtype    = strdup("MikMod UniFormat 3.0");
+
+    _mm_fseek(modfp,5,SEEK_SET);    // skip the header
+    
+    of.flags      = _mm_read_M_UWORD(modfp);
+    of.numchn     = _mm_read_UBYTE(modfp);
+    of.numvoices  = _mm_read_UBYTE(modfp);
+    of.numpos     = _mm_read_M_UWORD(modfp);
+    of.numpat     = _mm_read_M_UWORD(modfp);
+    of.numtrk     = _mm_read_M_UWORD(modfp);
+    of.numins     = _mm_read_M_UWORD(modfp);
+    of.numsmp     = _mm_read_M_UWORD(modfp);
+    of.reppos     = _mm_read_M_UWORD(modfp);
+    of.initspeed  = _mm_read_UBYTE(modfp);
+    of.inittempo  = _mm_read_UBYTE(modfp);
+    of.initvolume = _mm_read_UBYTE(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    of.songname = StringRead(modfp);
+    of.composer = StringRead(modfp);
+    of.comment  = StringRead(modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    if(!AllocSamples())            return 0;
+    if(!AllocTracks())             return 0;
+    if(!AllocPatterns())           return 0;
+    if(!AllocPositions(of.numpos)) return 0;
+
+    _mm_read_UBYTES(of.positions,of.numpos,modfp);
+    _mm_read_M_UWORDS(of.panning,of.numchn,modfp);
+    _mm_read_UBYTES(of.chanvol,of.numchn,modfp);
+
+
+    // Load sample headers
+
+    s = of.samples;
+    for(v=0; v<of.numsmp; v++, s++)
+    {   s->flags      = _mm_read_M_UWORD(modfp);
+        s->speed      = _mm_read_M_ULONG(modfp);
+        s->volume     = _mm_read_UBYTE(modfp);
+        s->panning    = _mm_read_M_UWORD(modfp);
+        s->length     = _mm_read_M_ULONG(modfp);
+        s->loopstart  = _mm_read_M_ULONG(modfp);
+        s->loopend    = _mm_read_M_ULONG(modfp);
+        s->susbegin   = _mm_read_M_ULONG(modfp);
+        s->susend     = _mm_read_M_ULONG(modfp);
+
+        s->globvol    = _mm_read_UBYTE(modfp);
+        s->vibflags   = _mm_read_UBYTE(modfp);
+        s->vibtype    = _mm_read_UBYTE(modfp);
+        s->vibsweep   = _mm_read_UBYTE(modfp);
+        s->vibdepth   = _mm_read_UBYTE(modfp);
+        s->vibrate    = _mm_read_UBYTE(modfp);
+        s->samplename = StringRead(modfp);
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_HEADER;
+            return 0;
+        }
+    }
+
+    // Load instruments
+
+    if(of.flags & UF_INST)
+    {   if(!AllocInstruments()) return 0;
+        i = of.instruments;
+
+        for(v=0; v<of.numins; v++, i++)
+        {   i->flags        = _mm_read_UBYTE(modfp);
+            i->nnatype      = _mm_read_UBYTE(modfp);
+            i->dca          = _mm_read_UBYTE(modfp);
+            i->dct          = _mm_read_UBYTE(modfp);
+            i->globvol      = _mm_read_UBYTE(modfp);
+            i->panning      = _mm_read_M_UWORD(modfp);
+            i->pitpansep    = _mm_read_UBYTE(modfp);
+            i->pitpancenter = _mm_read_UBYTE(modfp);
+            i->rvolvar      = _mm_read_UBYTE(modfp);
+            i->rpanvar      = _mm_read_UBYTE(modfp);
+
+            i->volfade      = _mm_read_M_UWORD(modfp);
+    
+            i->volflg       = _mm_read_UBYTE(modfp);
+            i->volpts       = _mm_read_UBYTE(modfp);
+            i->volsusbeg    = _mm_read_UBYTE(modfp);
+            i->volsusend    = _mm_read_UBYTE(modfp);
+            i->volbeg       = _mm_read_UBYTE(modfp);
+            i->volend       = _mm_read_UBYTE(modfp);
+    
+            for(w=0; w<i->volpts; w++)
+            {   i->volenv[w].pos = _mm_read_M_SWORD(modfp);
+                i->volenv[w].val = _mm_read_M_SWORD(modfp);
+            }
+    
+            i->panflg    = _mm_read_UBYTE(modfp);
+            i->panpts    = _mm_read_UBYTE(modfp);
+            i->pansusbeg = _mm_read_UBYTE(modfp);
+            i->pansusend = _mm_read_UBYTE(modfp);
+            i->panbeg    = _mm_read_UBYTE(modfp);
+            i->panend    = _mm_read_UBYTE(modfp);
+    
+            for(w=0; w<i->panpts; w++)
+            {   i->panenv[w].pos = _mm_read_M_SWORD(modfp);
+                i->panenv[w].val = _mm_read_M_SWORD(modfp);
+            }
+    
+            i->pitflg    = _mm_read_UBYTE(modfp);
+            i->pitpts    = _mm_read_UBYTE(modfp);
+            i->pitsusbeg = _mm_read_UBYTE(modfp);
+            i->pitsusend = _mm_read_UBYTE(modfp);
+            i->pitbeg    = _mm_read_UBYTE(modfp);
+            i->pitend    = _mm_read_UBYTE(modfp);
+    
+            for(w=0; w<i->pitpts; w++)
+            {   i->pitenv[w].pos = _mm_read_M_SWORD(modfp);
+                i->pitenv[w].val = _mm_read_M_SWORD(modfp);
+            }
+    
+            _mm_read_UBYTES(i->samplenumber, 120, modfp);
+            _mm_read_UBYTES(i->samplenote, 120, modfp);
+    
+            i->insname = StringRead(modfp);
+
+            if(feof(modfp))
+            {   _mm_errno = MMERR_LOADING_HEADER;
+                return 0;
+            }
+        }
+    }
+
+    // Read patterns
+
+    _mm_read_M_UWORDS(of.pattrows,of.numpat,modfp);
+    _mm_read_M_UWORDS(of.patterns,of.numpat*of.numchn,modfp);
+
+    // Read tracks
+
+    for(t=0; t<of.numtrk; t++)
+        of.tracks[t] = TrkRead();
+
+    return 1;
+}
+
+
+CHAR *UNI_LoadTitle(void)
+{
+    _mm_fseek(modfp,24,SEEK_SET);
+    return(StringRead(modfp));
+}
+
+
+MLOADER load_uni =
+{   NULL,
+    "UNI",
+    "Portable UniFormat 3.0 Loader",
+    UNI_Init,
+    UNI_Test,
+    UNI_Load,
+    UNI_Cleanup,
+    UNI_LoadTitle
+};
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_wav.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,97 @@
+/*
+
+ Name:  LOAD_WAV.C
+
+ Description:
+ WAV Streaming Audio Loader / Player
+
+ Portability:
+ All compilers -- All systems (hopefully)
+
+*/
+
+#include "mikmod.h"
+#include <string.h>
+
+
+typedef struct
+{   CHAR  rID[4];
+    ULONG rLen;
+    CHAR  wID[4];
+    CHAR  fID[4];
+    ULONG fLen;
+    UWORD wFormatTag;
+    UWORD nChannels;
+    ULONG nSamplesPerSec;
+    ULONG nAvgBytesPerSec;
+    UWORD nBlockAlign;
+    UWORD nFormatSpecific;
+} WAV;
+
+
+BOOL WAV_Load(void)
+{
+    SAMPLE     *si;
+    static WAV  wh;
+    static CHAR dID[4];
+
+    // read wav header
+
+    _mm_read_string(wh.rID,4,stream_fp);
+    wh.rLen            = _mm_read_I_ULONG(stream_fp);
+    _mm_read_string(wh.wID,4,stream_fp);
+    _mm_read_string(wh.fID,4,stream_fp);
+    wh.fLen            = _mm_read_I_ULONG(stream_fp);
+    wh.wFormatTag      = _mm_read_I_UWORD(stream_fp);
+    wh.nChannels       = _mm_read_I_UWORD(stream_fp);
+    wh.nSamplesPerSec  = _mm_read_I_ULONG(stream_fp);
+    wh.nAvgBytesPerSec = _mm_read_I_ULONG(stream_fp);
+    wh.nBlockAlign     = _mm_read_I_UWORD(stream_fp);
+    wh.nFormatSpecific = _mm_read_I_UWORD(stream_fp);
+
+    // check it
+
+    if( feof(stream_fp) ||
+        memcmp(wh.rID,"RIFF",4) ||
+        memcmp(wh.wID,"WAVE",4) ||
+        memcmp(wh.fID,"fmt ",4) )
+    {
+        _mm_errno = MMERR_UNKNOWN_WAVE_TYPE;
+        return NULL;
+    }
+
+    // skip other crap
+
+    _mm_fseek(stream_fp,wh.fLen-16,SEEK_CUR);
+    _mm_read_string(dID,4,stream_fp);
+
+    if( memcmp(dID,"data",4) )
+    {   _mm_errno = MMERR_UNKNOWN_WAVE_TYPE;
+        return NULL;
+    }
+
+    if(wh.nChannels > 1)
+    {   _mm_errno = MMERR_UNKNOWN_WAVE_TYPE;
+        return NULL;
+    }
+
+//  printf("wFormatTag: %x\n",wh.wFormatTag);
+//  printf("blockalign: %x\n",wh.nBlockAlign);
+//  prinff("nFormatSpc: %x\n",wh.nFormatSpecific);
+
+    if((si=(SAMPLE *)_mm_calloc(1,sizeof(SAMPLE)))==NULL) return NULL;
+
+    si->speed  = wh.nSamplesPerSec/wh.nChannels;
+    si->volume = 64;
+
+    si->length = _mm_read_I_ULONG(stream_fp);
+
+    if(wh.nBlockAlign==2)
+    {   si->flags = SF_16BITS | SF_SIGNED;
+        si->length>>=1;
+    }
+
+    return 0;
+}
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/load_xm.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,679 @@
+/*
+
+ Name: LOAD_XM.C
+
+ Description:
+ Fasttracker (XM) module loader
+
+ Portability:
+ All systems - all compilers (hopefully)
+
+ If this module is found to not be portable to any particular platform,
+ please contact Jake Stine at dracoirs@epix.net (see MIKMOD.TXT for
+ more information on contacting the author).
+
+*/
+
+#include <string.h>
+#include "mikmod.h"
+
+/**************************************************************************
+**************************************************************************/
+
+
+typedef struct XMHEADER
+{   CHAR  id[17];                   // ID text: 'Extended module: '
+    CHAR  songname[21];             // Module name, padded with zeroes and 0x1a at the end
+    CHAR  trackername[20];          // Tracker name
+    UWORD version;                  // (word) Version number, hi-byte major and low-byte minor
+    ULONG headersize;               // Header size
+    UWORD songlength;               // (word) Song length (in patten order table)
+    UWORD restart;                  // (word) Restart position
+    UWORD numchn;                   // (word) Number of channels (2,4,6,8,10,...,32)
+    UWORD numpat;                   // (word) Number of patterns (max 256)
+    UWORD numins;                   // (word) Number of instruments (max 128)
+    UWORD flags;                    // (word) Flags: bit 0: 0 = Amiga frequency table (see below) 1 = Linear frequency table
+    UWORD tempo;                    // (word) Default tempo
+    UWORD bpm;                      // (word) Default BPM
+    UBYTE orders[256];              // (byte) Pattern order table 
+} XMHEADER;
+
+
+typedef struct XMINSTHEADER
+{   ULONG size;                     // (dword) Instrument size
+    CHAR  name[22];                 // (char) Instrument name
+    UBYTE type;                     // (byte) Instrument type (always 0)
+    UWORD numsmp;                   // (word) Number of samples in instrument
+    ULONG ssize;                    //
+} XMINSTHEADER;
+
+
+typedef struct XMPATCHHEADER
+{   UBYTE what[96];         // (byte) Sample number for all notes
+    UWORD volenv[24];       // (byte) Points for volume envelope
+    UWORD panenv[24];       // (byte) Points for panning envelope
+    UBYTE volpts;           // (byte) Number of volume points
+    UBYTE panpts;           // (byte) Number of panning points
+    UBYTE volsus;           // (byte) Volume sustain point
+    UBYTE volbeg;           // (byte) Volume loop start point
+    UBYTE volend;           // (byte) Volume loop end point
+    UBYTE pansus;           // (byte) Panning sustain point
+    UBYTE panbeg;           // (byte) Panning loop start point
+    UBYTE panend;           // (byte) Panning loop end point
+    UBYTE volflg;           // (byte) Volume type: bit 0: On; 1: Sustain; 2: Loop
+    UBYTE panflg;           // (byte) Panning type: bit 0: On; 1: Sustain; 2: Loop
+    UBYTE vibflg;           // (byte) Vibrato type
+    UBYTE vibsweep;         // (byte) Vibrato sweep
+    UBYTE vibdepth;         // (byte) Vibrato depth
+    UBYTE vibrate;          // (byte) Vibrato rate
+    UWORD volfade;          // (word) Volume fadeout
+    UWORD reserved[11];     // (word) Reserved
+} XMPATCHHEADER;
+
+
+typedef struct XMWAVHEADER
+{   ULONG length;           // (dword) Sample length
+    ULONG loopstart;        // (dword) Sample loop start
+    ULONG looplength;       // (dword) Sample loop length
+    UBYTE volume;           // (byte) Volume 
+    SBYTE finetune;         // (byte) Finetune (signed byte -128..+127)
+    UBYTE type;             // (byte) Type: Bit 0-1: 0 = No loop, 1 = Forward loop,
+                            //                       2 = Ping-pong loop;
+                            //                    4: 16-bit sampledata
+    UBYTE panning;          // (byte) Panning (0-255)
+    SBYTE relnote;          // (byte) Relative note number (signed byte)
+    UBYTE reserved;         // (byte) Reserved
+    CHAR  samplename[22];   // (char) Sample name
+
+    UBYTE vibtype;          // (byte) Vibrato type
+    UBYTE vibsweep;         // (byte) Vibrato sweep
+    UBYTE vibdepth;         // (byte) Vibrato depth
+    UBYTE vibrate;          // (byte) Vibrato rate
+} XMWAVHEADER;
+
+
+typedef struct XMPATHEADE
+{   ULONG size;                     // (dword) Pattern header length 
+    UBYTE packing;                  // (byte) Packing type (always 0)
+    UWORD numrows;                  // (word) Number of rows in pattern (1..256)
+    UWORD packsize;                 // (word) Packed patterndata size
+} XMPATHEADER;
+
+typedef struct MTMNOTE
+{    UBYTE a,b,c;
+} MTMNOTE;
+
+
+typedef struct XMNOTE
+{    UBYTE note,ins,vol,eff,dat;
+}XMNOTE;
+
+/**************************************************************************
+**************************************************************************/
+
+static XMNOTE *xmpat = NULL;
+static XMHEADER *mh = NULL;
+
+BOOL XM_Test(void)
+{
+    UBYTE id[17];
+    
+    if(!_mm_read_UBYTES(id,17,modfp)) return 0;
+    if(!memcmp(id,"Extended Module: ",17)) return 1;
+    return 0;
+}
+
+
+BOOL XM_Init(void)
+{
+    if(!(mh=(XMHEADER *)_mm_calloc(1,sizeof(XMHEADER)))) return 0;
+    return 1;
+}
+
+
+void XM_Cleanup(void)
+{
+    if(mh!=NULL) free(mh);
+    mh = NULL;
+}
+
+
+void XM_ReadNote(XMNOTE *n)
+{
+    UBYTE cmp;
+
+    memset(n,0,sizeof(XMNOTE));
+
+    cmp = _mm_read_UBYTE(modfp);
+
+    if(cmp&0x80)
+    {   if(cmp&1)  n->note = _mm_read_UBYTE(modfp);
+        if(cmp&2)  n->ins  = _mm_read_UBYTE(modfp);
+        if(cmp&4)  n->vol  = _mm_read_UBYTE(modfp);
+        if(cmp&8)  n->eff  = _mm_read_UBYTE(modfp);
+        if(cmp&16) n->dat  = _mm_read_UBYTE(modfp);
+    }
+    else
+    {   n->note = cmp;
+        n->ins  = _mm_read_UBYTE(modfp);
+        n->vol  = _mm_read_UBYTE(modfp);
+        n->eff  = _mm_read_UBYTE(modfp);
+        n->dat  = _mm_read_UBYTE(modfp);
+    }
+}
+
+
+UBYTE *XM_Convert(XMNOTE *xmtrack,UWORD rows)
+{
+    int t;
+    UBYTE note,ins,vol,eff,dat;
+
+    UniReset();
+
+    for(t=0; t<rows; t++)
+    {   note = xmtrack->note;
+        ins  = xmtrack->ins;
+        vol  = xmtrack->vol;
+        eff  = xmtrack->eff;
+        dat  = xmtrack->dat;
+
+        if(note!=0)
+        {  if(note==97) 
+           {   UniWrite(UNI_KEYFADE);
+               UniWrite(0);
+           } else
+               UniNote(note-1);
+        }
+
+        if(ins!=0) UniInstrument(ins-1);
+
+        switch(vol>>4)
+        {
+            case 0x6:                   // volslide down
+                if(vol&0xf)
+                {   UniWrite(UNI_XMEFFECTA);
+                    UniWrite(vol&0xf);
+                }
+                break;
+
+            case 0x7:                   // volslide up
+                if(vol&0xf)
+                {   UniWrite(UNI_XMEFFECTA);
+                    UniWrite(vol<<4);
+                }
+                break;
+
+            // volume-row fine volume slide is compatible with protracker
+            // EBx and EAx effects i.e. a zero nibble means DO NOT SLIDE, as
+            // opposed to 'take the last sliding value'.
+
+            case 0x8:                       // finevol down
+                UniPTEffect(0xe,0xb0 | (vol&0xf));
+                break;
+
+            case 0x9:                       // finevol up
+                UniPTEffect(0xe,0xa0 | (vol&0xf));
+                break;
+
+            case 0xa:                       // set vibrato speed
+                UniPTEffect(0x4,vol<<4);
+                break;
+
+            case 0xb:                       // vibrato
+                UniPTEffect(0x4,vol&0xf);
+                break;
+
+            case 0xc:                       // set panning
+                UniPTEffect(0x8,vol<<4);
+                break;
+
+            case 0xd:                       // panning slide left
+                // only slide when data nibble not zero:
+
+                if(vol&0xf)
+                {   UniWrite(UNI_XMEFFECTP);
+                    UniWrite(vol&0xf);
+                }
+                break;
+
+            case 0xe:                       // panning slide right
+                // only slide when data nibble not zero:
+
+                if(vol&0xf)
+                {   UniWrite(UNI_XMEFFECTP);
+                    UniWrite(vol<<4);
+                }
+             break;
+
+            case 0xf:                       // tone porta
+                UniPTEffect(0x3,vol<<4);
+            break;
+
+            default:
+                if(vol>=0x10 && vol<=0x50)
+                    UniPTEffect(0xc,vol-0x10);
+        }
+
+        switch(eff)
+        {
+            case 0x4:                       // Effect 4: Vibrato
+                UniWrite(UNI_XMEFFECT4);
+                UniWrite(dat);
+            break;
+
+            case 0xa:
+                UniWrite(UNI_XMEFFECTA);
+                UniWrite(dat);
+            break;
+
+            case 0xe:
+                switch(dat>>4)
+                {  case 0x1:      // XM fine porta up
+                      UniWrite(UNI_XMEFFECTE1);
+                      UniWrite(dat&0xf);
+                   break;
+
+                   case 0x2:      // XM fine porta down
+                      UniWrite(UNI_XMEFFECTE2);
+                      UniWrite(dat&0xf);
+                   break;
+
+                   case 0xa:      // XM fine volume up
+                      UniWrite(UNI_XMEFFECTEA);
+                      UniWrite(dat&0xf);
+                   break;
+
+                   case 0xb:      // XM fine volume down
+                      UniWrite(UNI_XMEFFECTEB);
+                      UniWrite(dat&0xf);
+                   break;
+
+                   default:
+                      UniPTEffect(0x0e,dat);
+                }
+            break;
+
+            case 'G'-55:                    // G - set global volume
+                if(dat>64) dat = 64;
+                UniWrite(UNI_XMEFFECTG);
+                UniWrite(dat);
+                break;
+
+            case 'H'-55:                    // H - global volume slide
+                UniWrite(UNI_XMEFFECTH);
+                UniWrite(dat);
+                break;
+
+            case 'K'-55:                    // K - keyOff and KeyFade
+                UniWrite(UNI_KEYFADE);
+                UniWrite(dat);
+                break;
+
+            case 'L'-55:                    // L - set envelope position
+                UniWrite(UNI_XMEFFECTL);
+                UniWrite(dat);
+                break;
+
+            case 'P'-55:                    // P - panning slide
+                UniWrite(UNI_XMEFFECTP);
+                UniWrite(dat);
+                break;
+
+            case 'R'-55:                    // R - multi retrig note
+                UniWrite(UNI_S3MEFFECTQ);
+                UniWrite(dat);
+                break;
+
+            case 'T'-55:                    // T - Tremor !! (== S3M effect I)
+                UniWrite(UNI_S3MEFFECTI);
+                UniWrite(dat);
+                break;
+
+            case 'X'-55:
+                if((dat>>4) == 1)           // X1 - Extra Fine Porta up
+                {  UniWrite(UNI_XMEFFECTX1);
+                   UniWrite(dat & 0xf);
+                } else if((dat>>4) == 2)    // X2 - Extra Fine Porta down
+                {  UniWrite(UNI_XMEFFECTX2);
+                   UniWrite(dat & 0xf);
+                }
+            break;
+
+            default:
+                if(eff <= 0xf)
+                {   // Convert pattern jump from Dec to Hex
+                    if(eff == 0xd)
+                        dat = (((dat&0xf0)>>4)*10)+(dat&0xf);
+                    UniPTEffect(eff,dat);
+                }
+            break;
+        }
+
+        UniNewline();
+        xmtrack++;
+    }
+    return UniDup();
+}
+
+
+
+BOOL XM_Load(void)
+{
+    INSTRUMENT *d;
+    SAMPLE     *q;
+    XMWAVHEADER *wh,*s;
+    int   t,u,v,p,numtrk;
+    long  next;
+    ULONG nextwav[256];
+    BOOL  dummypat=0;
+ 
+    // try to read module header
+
+    _mm_read_string(mh->id,17,modfp);
+    _mm_read_string(mh->songname,21,modfp);
+    _mm_read_string(mh->trackername,20,modfp);
+    mh->version     =_mm_read_I_UWORD(modfp);
+    mh->headersize  =_mm_read_I_ULONG(modfp);
+    mh->songlength  =_mm_read_I_UWORD(modfp);
+    mh->restart     =_mm_read_I_UWORD(modfp);
+    mh->numchn      =_mm_read_I_UWORD(modfp);
+    mh->numpat      =_mm_read_I_UWORD(modfp);
+    mh->numins      =_mm_read_I_UWORD(modfp);
+    mh->flags       =_mm_read_I_UWORD(modfp);
+    mh->tempo       =_mm_read_I_UWORD(modfp);
+    mh->bpm         =_mm_read_I_UWORD(modfp);
+    _mm_read_UBYTES(mh->orders,256,modfp);
+
+    if(feof(modfp))
+    {   _mm_errno = MMERR_LOADING_HEADER;
+        return 0;
+    }
+
+    // set module variables
+    of.initspeed = mh->tempo;         
+    of.inittempo = mh->bpm;
+    of.modtype   = DupStr(mh->trackername,20);
+    of.numchn    = mh->numchn;
+    of.numpat    = mh->numpat;
+    of.numtrk    = (UWORD)of.numpat*of.numchn;   // get number of channels
+    of.songname  = DupStr(mh->songname,20);      // make a cstr of songname
+    of.numpos    = mh->songlength;               // copy the songlength
+    of.reppos    = mh->restart;
+    of.numins    = mh->numins;
+    of.flags |= UF_XMPERIODS | UF_INST;
+    if(mh->flags&1) of.flags |= UF_LINEAR;
+
+    memset(of.chanvol,64,of.numchn);             // store channel volumes
+
+    if(!AllocPositions(of.numpos+3)) return 0;
+    for(t=0; t<of.numpos; t++)
+        of.positions[t] = mh->orders[t];
+
+/*
+   WHY THIS CODE HERE?? I CAN'T REMEMBER!
+   
+   Well, I do know why, mikmak!  Seems that FT2 doesn't always count blank 
+   patterns AT ALL if they are at the END of the song.  So, we have to check
+   for any patter numbers in the order list greater than the number of pat-
+   terns total.  If one or more is found, we set it equal to the pattern total
+   and make a dummy pattern to accomidate for the discrepency!
+*/
+
+    for(t=0; t<of.numpos; t++)
+    {   if(of.positions[t] > of.numpat)
+        {  of.positions[t] = of.numpat;
+           dummypat = 1;
+        }
+    }      
+
+    if(dummypat) { of.numpat++; of.numtrk+=of.numchn; }
+
+    if(!AllocTracks()) return 0;
+    if(!AllocPatterns()) return 0;
+
+    numtrk = 0;
+    for(t=0; t<mh->numpat; t++)
+    {   XMPATHEADER ph;
+
+        ph.size     =_mm_read_I_ULONG(modfp);
+        ph.packing  =_mm_read_UBYTE(modfp);
+        ph.numrows  =_mm_read_I_UWORD(modfp);
+        ph.packsize =_mm_read_I_UWORD(modfp);
+
+        of.pattrows[t] = ph.numrows;
+
+        //  Gr8.. when packsize is 0, don't try to load a pattern.. it's empty.
+        //  This bug was discovered thanks to Khyron's module..
+
+        if(!(xmpat=(XMNOTE *)_mm_calloc(ph.numrows*of.numchn,sizeof(XMNOTE)))) return 0;
+
+        if(ph.packsize>0)
+        {   for(u=0; u<ph.numrows; u++)
+            {   for(v=0; v<of.numchn; v++)
+                    XM_ReadNote(&xmpat[(v*ph.numrows)+u]);
+            }
+        }
+
+        if(feof(modfp))
+        {   _mm_errno = MMERR_LOADING_PATTERN;
+            return 0;
+        }
+
+        for(v=0; v<of.numchn; v++)                                           
+           of.tracks[numtrk++] = XM_Convert(&xmpat[v*ph.numrows],ph.numrows);
+
+        free(xmpat);
+    }
+                                                          
+    if(dummypat)
+    {  of.pattrows[t] = 64;
+       if(!(xmpat=(XMNOTE *)_mm_calloc(64*of.numchn,sizeof(XMNOTE)))) return 0;
+       for(v=0; v<of.numchn; v++)
+           of.tracks[numtrk++] = XM_Convert(&xmpat[v*64],64);
+       free(xmpat);
+    }
+
+    if(!AllocInstruments()) return 0;
+    if((wh = (XMWAVHEADER *)_mm_calloc(256,sizeof(XMWAVHEADER))) == NULL) return 0;
+    d = of.instruments;
+    s = wh;
+
+
+    for(t=0; t<of.numins; t++)
+    {   XMINSTHEADER ih;
+        int          headend;
+
+        memset(d->samplenumber,255,120);
+
+        // read instrument header
+
+        headend     = _mm_ftell(modfp);
+        ih.size     = _mm_read_I_ULONG(modfp);
+        headend    += ih.size;
+        _mm_read_string(ih.name, 22, modfp);
+        ih.type     = _mm_read_UBYTE(modfp);
+        ih.numsmp   = _mm_read_I_UWORD(modfp);
+        d->insname  = DupStr(ih.name,22);
+
+        if(ih.size > 29)
+        {   ih.ssize    = _mm_read_I_ULONG(modfp);
+            if(ih.numsmp > 0)
+            {   XMPATCHHEADER pth;
+    
+                _mm_read_UBYTES (pth.what, 96, modfp);
+                _mm_read_I_UWORDS (pth.volenv, 24, modfp);
+                _mm_read_I_UWORDS (pth.panenv, 24, modfp);
+                pth.volpts      =  _mm_read_UBYTE(modfp);
+                pth.panpts      =  _mm_read_UBYTE(modfp);
+                pth.volsus      =  _mm_read_UBYTE(modfp);
+                pth.volbeg      =  _mm_read_UBYTE(modfp);
+                pth.volend      =  _mm_read_UBYTE(modfp);
+                pth.pansus      =  _mm_read_UBYTE(modfp);
+                pth.panbeg      =  _mm_read_UBYTE(modfp);
+                pth.panend      =  _mm_read_UBYTE(modfp);
+                pth.volflg      =  _mm_read_UBYTE(modfp);
+                pth.panflg      =  _mm_read_UBYTE(modfp);
+                pth.vibflg      =  _mm_read_UBYTE(modfp);
+                pth.vibsweep    =  _mm_read_UBYTE(modfp);
+                pth.vibdepth    =  _mm_read_UBYTE(modfp);
+                pth.vibrate     =  _mm_read_UBYTE(modfp);
+                pth.volfade     =  _mm_read_I_UWORD(modfp);
+    
+                // read the remainder of the header
+                for(u=headend-_mm_ftell(modfp); u; u--)  _mm_read_UBYTE(modfp);
+    
+                if(feof(modfp))
+                {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+                    return 0;
+                }
+    
+                for(u=0; u<96; u++)         
+                   d->samplenumber[u] = pth.what[u] + of.numsmp;
+    
+                d->volfade = pth.volfade;
+    
+                memcpy(d->volenv,pth.volenv,24);
+                if(pth.volflg & 1)  d->volflg |= EF_ON;
+                if(pth.volflg & 2)  d->volflg |= EF_SUSTAIN;
+                if(pth.volflg & 4)  d->volflg |= EF_LOOP;
+                d->volsusbeg = d->volsusend = pth.volsus;
+                d->volbeg    = pth.volbeg;
+                d->volend    = pth.volend;
+                d->volpts    = pth.volpts;
+    
+                // scale volume envelope:
+    
+                for(p=0; p<12; p++)
+                    d->volenv[p].val <<= 2;
+    
+                if((d->volflg & EF_ON) && (d->volpts < 2))
+                    d->volflg &= ~EF_ON;
+    
+                memcpy(d->panenv,pth.panenv,24);
+                d->panflg    = pth.panflg;
+                d->pansusbeg = d->pansusend = pth.pansus;
+                d->panbeg    = pth.panbeg;
+                d->panend    = pth.panend;
+                d->panpts    = pth.panpts;
+    
+                // scale panning envelope:
+    
+                for(p=0; p<12; p++)
+                    d->panenv[p].val <<= 2;
+                if((d->panflg & EF_ON) && (d->panpts < 2))
+                    d->panflg &= ~EF_ON;
+    
+                next = 0;
+    
+                //  Samples are stored outside the instrument struct now, so we have
+                //  to load them all into a temp area, count the of.numsmp along the
+                //  way and then do an AllocSamples() and move everything over 
+    
+                for(u=0; u<ih.numsmp; u++,s++)
+                {   s->length       =_mm_read_I_ULONG (modfp);
+                    s->loopstart    =_mm_read_I_ULONG (modfp);
+                    s->looplength   =_mm_read_I_ULONG (modfp);
+                    s->volume       =_mm_read_UBYTE (modfp);
+                    s->finetune     =_mm_read_SBYTE (modfp);
+                    s->type         =_mm_read_UBYTE (modfp);
+                    s->panning      =_mm_read_UBYTE (modfp);
+                    s->relnote      =_mm_read_SBYTE (modfp);
+                    s->vibtype      = pth.vibflg;
+                    s->vibsweep     = pth.vibsweep;
+                    s->vibdepth     = pth.vibdepth*4;
+                    s->vibrate      = pth.vibrate;
+    
+                    s->reserved =_mm_read_UBYTE (modfp);
+                    _mm_read_string(s->samplename, 22, modfp);
+    
+                    nextwav[of.numsmp+u] = next;
+                    next += s->length;
+    
+                    if(feof(modfp))
+                    {   _mm_errno = MMERR_LOADING_SAMPLEINFO;
+                        return 0;
+                    }
+                }
+    
+                for(u=0; u<ih.numsmp; u++) nextwav[of.numsmp++] += _mm_ftell(modfp);
+                _mm_fseek(modfp,next,SEEK_CUR);
+            }
+        }
+
+        d++;
+    }
+
+    if(!AllocSamples()) return 0;
+    q = of.samples;
+    s = wh;
+
+    for(u=0; u<of.numsmp; u++,q++,s++)
+    {   q->samplename   = DupStr(s->samplename,22);
+        q->length       = s->length;
+        q->loopstart    = s->loopstart;
+        q->loopend      = s->loopstart+s->looplength;
+        q->volume       = s->volume;
+        q->speed        = s->finetune+128;
+        q->panning      = s->panning;
+        q->seekpos      = nextwav[u];
+        q->vibtype      = s->vibtype;
+        q->vibsweep     = s->vibsweep;
+        q->vibdepth     = s->vibdepth;
+        q->vibrate      = s->vibrate;
+
+        if(s->type & 0x10)
+        {   q->length    >>= 1;
+            q->loopstart >>= 1;
+            q->loopend   >>= 1;
+        }
+
+        q->flags|=SF_OWNPAN;
+        if(s->type&0x3) q->flags|=SF_LOOP;
+        if(s->type&0x2) q->flags|=SF_BIDI;
+
+        if(s->type&0x10) q->flags|=SF_16BITS;
+        q->flags|=SF_DELTA;
+        q->flags|=SF_SIGNED;
+    }
+
+    d = of.instruments;
+    s = wh;
+    for(u=0; u<of.numins; u++, d++)
+    {  /*for(t=0; t<3; t++)
+          if((s[d->samplenumber[t]].relnote / 12) > )
+          {  s[d->samplenumber[t]].relnote -= 12;
+             of.samples[d->samplenumber[t]].speed <<= 1;
+          }
+       */
+       for(t=0; t<96; t++)
+          d->samplenote[t] = (d->samplenumber[t]==of.numsmp) ? 255 : (t+s[d->samplenumber[t]].relnote);
+    }
+
+    free(wh);
+    return 1;
+}
+
+
+
+CHAR *XM_LoadTitle(void)
+{
+    CHAR s[21];
+
+    _mm_fseek(modfp,17,SEEK_SET);
+    if(!fread(s,21,1,modfp)) return NULL;
+  
+    return(DupStr(s,21));
+}
+
+
+
+MLOADER load_xm =
+{   NULL,
+    "XM",
+    "Portable XM loader v0.5",
+    XM_Init,
+    XM_Test,
+    XM_Load,
+    XM_Cleanup,
+    XM_LoadTitle
+};
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/mloader.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,399 @@
+/*
+
+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;
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/stream.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,134 @@
+/*
+ File: STREAM.C
+
+ Description:
+ Streaming Audio module for MikMod.
+
+ Portability:
+ All compilers - All systems
+
+*/
+
+
+#include "mikmod.h"
+
+
+int   stream_bufsize = 100;  // in 1/100th seconds
+
+static BOOL    is_streaming = 0;
+static MSTREAM *firststream = NULL;
+
+static BOOL active;
+
+FILE  *stream_fp;
+SLONG stream_seekpos;
+SLONG stream_reppos;
+
+void MikMod_RegisterStream(MSTREAM *stream)
+{
+    MSTREAM *cruise = firststream;
+
+    if(cruise!=NULL)
+    {   while(cruise->next!=NULL)  cruise = cruise->next;
+        cruise->next = stream;
+    } else
+        firststream = stream;
+}
+
+
+BOOL Stream_PlayFN(CHAR *filename)
+{
+    return 0;
+}
+
+
+BOOL Stream_PlayFP(FILE *fp)
+{
+    BOOL    ok;
+    MSTREAM *l;
+
+    stream_fp = fp;
+    _mm_errno = 0;
+    _mm_critical = 0;
+
+    _mm_iobase_setcur(stream_fp);
+
+    // Try to find a loader that recognizes the stream
+
+    for(l=firststream; l!=NULL; l=l->next)
+    {   _mm_rewind(stream_fp);
+        if(l->Test()) break;
+    }
+
+    if(l==NULL)
+    {   _mm_errno = MMERR_NOT_A_STREAM;
+        if(_mm_errorhandler!=NULL) _mm_errorhandler();
+        _mm_iobase_revert();
+        return 1;
+    }
+
+    // init stream loader and load the header
+    if(l->Init())
+    {   _mm_rewind(stream_fp);
+        ok = l->Load();
+    }
+
+    // free loader allocations
+    l->Cleanup();
+
+    if(!ok)
+    {   _mm_iobase_revert();
+        return 1;
+    }
+
+    //loadbuf = _mm_malloc(8192);
+
+    _mm_iobase_revert();
+    active = 1;
+    return 0;
+}
+
+
+BOOL Stream_Init(ULONG speed, UWORD flags)
+
+// size = size of buffer in 1/100th seconds
+
+{
+    if(is_streaming) md_driver->StreamExit();
+    if(md_driver->StreamInit(speed,flags)) return 1;
+    is_streaming = 1;
+
+    return 0;
+}
+
+
+void Stream_Exit(void)
+{
+    if(is_streaming) md_driver->StreamExit();
+    is_streaming = 0;
+}
+
+
+static ULONG last = 0;
+
+void Stream_Update(void)
+{
+    ULONG curr;
+    ULONG todo;
+
+    curr = md_driver->StreamGetPosition();
+    if(curr==last) return;
+
+    if(curr>last)
+    {   todo = curr-last;
+        //SL_LoadStream(,vstream.infmt,vstream.flags,todo,stream_fp);
+        last += todo;
+        //if(last>=vstream.size) last = 0;
+    } else
+    {   //todo = vstream.size-last;
+        //SL_LoadStream(&vstream.buffer[last],vstream.infmt,vstream.flags,todo,stream_fp);
+        //SL_LoadStream(vstream.buffer,vstream.infmt,vstream.flags,curr,stream_fp);
+        last = curr;
+    }
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/playercode/virtch2.c	Fri Jan 23 16:05:08 1998 +0000
@@ -0,0 +1,927 @@
+/*
+
+Name:  VIRTCH2.C
+
+Description:
+ All-c sample mixing routines, using a 32 bits mixing buffer, interpolation,
+ and sample smoothing [improves sound quality and removes clicks].
+
+ Future Additions:
+   Low-Pass filter to remove annoying staticy buzz.
+
+
+C Mixer Portability:
+ All Systems -- All compilers.
+
+Assembly Mixer Portability:
+
+ MSDOS:  BC(?)   Watcom(y)   DJGPP(y)
+ Win95:  ?
+ Os2:    ?
+ Linux:  y
+
+ (y) - yes
+ (n) - no (not possible or not useful)
+ (?) - may be possible, but not tested
+
+*/
+
+#include <stddef.h>
+#include <string.h>
+#include "mikmod.h"
+
+
+// REVERBERATION : Larger numbers result in shorter reverb duration.
+//      Longer reverb durations can cause unwanted static and make the
+//      reverb sound more like an echo.
+
+#define REVERBERATION 28000l
+
+// SAMPLING_SHIFT : Specified the shift multiplier which controls by how
+//      much the mixing rate is multiplied while mixing.  Higher values
+//      can improve quality by smoothing the soudn and reducing pops and
+//      clicks.  Note, this is a shift value, so a value of 2 becomes a
+//      mixing-rate multiplier of 4, and a value of 3 = 8, etc.
+
+#define SAMPLING_SHIFT  2
+#define SAMPLING_FACTOR (SLONG)(1<<SAMPLING_SHIFT)
+
+
+// FRACBITS : the number of bits per integer devoted to the fractional
+//      part of the number.  This value HAS to be changed if the compiler
+//      does not support 64 bit integers.  If 32 bit integers are used,
+//      FRACBITS _must_ be 9 or smaller.
+
+#define FRACBITS 28
+#define FRACMASK ((1l<<FRACBITS)-1)
+
+#define TICKLSIZE 4096
+#define TICKWSIZE (TICKLSIZE*2)
+#define TICKBSIZE (TICKWSIZE*2)
+
+#ifndef MIN
+#define MIN(a,b) (((a)<(b)) ? (a) : (b))
+#endif
+
+#ifndef MAX
+#define MAX(a,b) (((a)>(b))?(a):(b))
+#endif
+
+
+typedef struct
+{   UBYTE  kick;                  // =1 -> sample has to be restarted
+    UBYTE  active;                // =1 -> sample is playing
+    UWORD  flags;                 // 16/8 bits looping/one-shot
+    SWORD  handle;                // identifies the sample
+    ULONG  start;                 // start index
+    ULONG  size;                  // samplesize
+    ULONG  reppos;                // loop start
+    ULONG  repend;                // loop end
+    ULONG  frq;                   // current frequency
+    UWORD  vol;                   // current volume
+    UWORD  pan;                   // current panning position
+    SDOUBLE  current;               // current index in the sample
+    SDOUBLE  increment;             // fixed-point increment value
+} VINFO;
+
+
+static SWORD   **Samples;
+
+static VINFO   *vinf = NULL;
+static VINFO   *vnf;
+static ULONG   samplesthatfit;
+static int     vc_memory, vc_softchn;
+static SDOUBLE idxsize,idxlpos,idxlend;
+static SLONG   TICKLEFT;
+static SLONG   *VC2_TICKBUF = NULL;
+static UWORD   vc_mode;
+static int     bitshift;           // Amplification shift (amount to decrease 32 bit mixing buffer by!)
+static SLONG   lvolsel, rvolsel;   // Volume factor .. range 0-255
+
+
+// Reverb control variables
+// ========================
+
+static int     RVc1, RVc2, RVc3, RVc4, RVc5, RVc6, RVc7, RVc8;
+static ULONG   RVRindex;
+
+// For Mono or Left Channel
+static SLONG  *RVbuf1 = NULL, *RVbuf2 = NULL, *RVbuf3 = NULL,
+              *RVbuf4 = NULL, *RVbuf5 = NULL, *RVbuf6 = NULL,
+              *RVbuf7 = NULL, *RVbuf8 = NULL;
+
+// For Stereo only (Right Channel)
+static SLONG  *RVbuf9 = NULL, *RVbuf10 = NULL, *RVbuf11 = NULL,
+              *RVbuf12 = NULL, *RVbuf13 = NULL, *RVbuf14 = NULL,
+              *RVbuf15 = NULL, *RVbuf16 = NULL;
+
+
+// ==============================================================
+//  16 bit sample mixers!
+
+static SDOUBLE MixStereoNormal(SWORD *srce, SLONG *dest, SDOUBLE index, SDOUBLE increment, ULONG todo)
+{
+    SWORD  sample;
+
+    for(; todo; todo--)
+    {   sample = (index & FRACBITS) ?
+                    (((srce[index >> FRACBITS] * (FRACBITS - (index & FRACBITS))) +
+                    (srce[(index >> FRACBITS)+1] * (index & FRACBITS))) / FRACBITS)
+                 :  srce[index >> FRACBITS];
+        index += increment;
+
+        *dest++ += lvolsel * sample;
+        *dest++ += rvolsel * sample;
+    }
+
+    return index;
+}
+
+
+static SDOUBLE MixStereoSurround(SWORD *srce, SLONG *dest, SDOUBLE index, SDOUBLE increment, ULONG todo)
+{
+    SWORD  sample;
+
+    for(dest--; todo; todo--)
+    {   sample = (index & FRACBITS) ?
+                    (((srce[index >> FRACBITS] * (FRACBITS - (index & FRACBITS))) +
+                    (srce[(index >> FRACBITS)+1] * (index & FRACBITS))) / FRACBITS)
+                 :  srce[index >> FRACBITS];
+        index += increment;
+
+        *dest++ += lvolsel * sample;
+        *dest++ -= lvolsel * sample;
+    }
+    
+    return index;
+}
+
+
+static SDOUBLE MixMonoNormal(SWORD *srce, SLONG *dest, SDOUBLE index, SDOUBLE increment, SLONG todo)
+{
+    SWORD  sample;
+    
+    for(; todo; todo--)
+    {   sample = (index & FRACBITS) ?
+                    (((srce[index >> FRACBITS] * (FRACBITS - (index & FRACBITS))) +
+                    (srce[(index >> FRACBITS)+1] * (index & FRACBITS))) / FRACBITS)
+                 :  srce[index >> FRACBITS];
+        index += increment;
+
+        *dest++ += lvolsel * sample;
+    }
+
+    return index;
+}
+
+
+static void (*Mix32to16)(SWORD *dste, SLONG *srce, SLONG count);
+static void (*Mix32to8)(SBYTE *dste, SLONG *srce, SLONG count);
+static void (*MixReverb)(SLONG *srce, SLONG count);
+
+
+static void MixReverb_Normal(SLONG *srce, SLONG count)
+{
+    SLONG        speedup;
+    int          ReverbPct;
+    unsigned int loc1, loc2, loc3, loc4,
+                 loc5, loc6, loc7, loc8;
+
+    ReverbPct = 63 + (md_reverb*4);
+
+    loc1 = RVRindex % RVc1;
+    loc2 = RVRindex % RVc2;
+    loc3 = RVRindex % RVc3;
+    loc4 = RVRindex % RVc4;
+    loc5 = RVRindex % RVc5;
+    loc6 = RVRindex % RVc6;
+    loc7 = RVRindex % RVc7;
+    loc8 = RVRindex % RVc8;
+
+    for(; count; count--)
+    {
+        // Compute the LEFT CHANNEL echo buffers!
+
+        speedup = *srce >> 3;
+
+        RVbuf1[loc1] = speedup + ((ReverbPct * RVbuf1[loc1]) / 128);
+        RVbuf2[loc2] = speedup + ((ReverbPct * RVbuf2[loc2]) / 128);
+        RVbuf3[loc3] = speedup + ((ReverbPct * RVbuf3[loc3]) / 128);
+        RVbuf4[loc4] = speedup + ((ReverbPct * RVbuf4[loc4]) / 128);
+        RVbuf5[loc5] = speedup + ((ReverbPct * RVbuf5[loc5]) / 128);
+        RVbuf6[loc6] = speedup + ((ReverbPct * RVbuf6[loc6]) / 128);
+        RVbuf7[loc7] = speedup + ((ReverbPct * RVbuf7[loc7]) / 128);
+        RVbuf8[loc8] = speedup + ((ReverbPct * RVbuf8[loc8]) / 128);
+
+        // Prepare to compute actual finalized data!
+
+        RVRindex++;
+        loc1 = RVRindex % RVc1;
+        loc2 = RVRindex % RVc2;
+        loc3 = RVRindex % RVc3;
+        loc4 = RVRindex % RVc4;
+        loc5 = RVRindex % RVc5;
+        loc6 = RVRindex % RVc6;
+        loc7 = RVRindex % RVc7;
+        loc8 = RVRindex % RVc8;
+
+        // Left Channel!
+        
+        *srce++ += (RVbuf1[loc1] - RVbuf2[loc2] + RVbuf3[loc3] - RVbuf4[loc4] +
+                    RVbuf5[loc5] - RVbuf6[loc6] + RVbuf7[loc7] - RVbuf8[loc8]);
+    }
+}
+
+
+static void MixReverb_Stereo(SLONG *srce, SLONG count)
+{
+    SLONG        speedup;
+    int          ReverbPct;
+    unsigned int loc1, loc2, loc3, loc4,
+                 loc5, loc6, loc7, loc8;
+
+    ReverbPct = 63 + (md_reverb*4);
+
+    loc1 = RVRindex % RVc1;
+    loc2 = RVRindex % RVc2;
+    loc3 = RVRindex % RVc3;
+    loc4 = RVRindex % RVc4;
+    loc5 = RVRindex % RVc5;
+    loc6 = RVRindex % RVc6;
+    loc7 = RVRindex % RVc7;
+    loc8 = RVRindex % RVc8;
+
+    for(; count; count--)
+    {
+        // Compute the LEFT CHANNEL echo buffers!
+
+        speedup = *srce >> 3;
+
+        RVbuf1[loc1] = speedup + ((ReverbPct * RVbuf1[loc1]) / 128);
+        RVbuf2[loc2] = speedup + ((ReverbPct * RVbuf2[loc2]) / 128);
+        RVbuf3[loc3] = speedup + ((ReverbPct * RVbuf3[loc3]) / 128);
+        RVbuf4[loc4] = speedup + ((ReverbPct * RVbuf4[loc4]) / 128);
+        RVbuf5[loc5] = speedup + ((ReverbPct * RVbuf5[loc5]) / 128);
+        RVbuf6[loc6] = speedup + ((ReverbPct * RVbuf6[loc6]) / 128);
+        RVbuf7[loc7] = speedup + ((ReverbPct * RVbuf7[loc7]) / 128);
+        RVbuf8[loc8] = speedup + ((ReverbPct * RVbuf8[loc8]) / 128);
+
+        // Compute the RIGHT CHANNEL echo buffers!
+        
+        speedup = srce[1] >> 3;
+
+        RVbuf9[loc1] = speedup + ((ReverbPct * RVbuf9[loc1]) / 128);
+        RVbuf10[loc2] = speedup + ((ReverbPct * RVbuf11[loc2]) / 128);
+        RVbuf11[loc3] = speedup + ((ReverbPct * RVbuf12[loc3]) / 128);
+        RVbuf12[loc4] = speedup + ((ReverbPct * RVbuf12[loc4]) / 128);
+        RVbuf13[loc5] = speedup + ((ReverbPct * RVbuf13[loc5]) / 128);
+        RVbuf14[loc6] = speedup + ((ReverbPct * RVbuf14[loc6]) / 128);
+        RVbuf15[loc7] = speedup + ((ReverbPct * RVbuf15[loc7]) / 128);
+        RVbuf16[loc8] = speedup + ((ReverbPct * RVbuf16[loc8]) / 128);
+
+        // Prepare to compute actual finalized data!
+
+        RVRindex++;
+        loc1 = RVRindex % RVc1;
+        loc2 = RVRindex % RVc2;
+        loc3 = RVRindex % RVc3;
+        loc4 = RVRindex % RVc4;
+        loc5 = RVRindex % RVc5;
+        loc6 = RVRindex % RVc6;
+        loc7 = RVRindex % RVc7;
+        loc8 = RVRindex % RVc8;
+
+        // Left Channel!
+        
+        *srce++ += (RVbuf1[loc1] - RVbuf2[loc2] + RVbuf3[loc3] - RVbuf4[loc4] +
+                    RVbuf5[loc5] - RVbuf6[loc6] + RVbuf7[loc7] - RVbuf8[loc8]);
+
+        // Right Channel!
+        
+        *srce++ += (RVbuf9[loc1] - RVbuf10[loc2] + RVbuf11[loc3] - RVbuf12[loc4] +
+                    RVbuf13[loc5] - RVbuf14[loc6] + RVbuf15[loc7] - RVbuf16[loc8]);
+    }
+}
+
+
+static void Mix32To16_Normal(SWORD *dste, SLONG *srce, SLONG count)
+{
+    SLONG    x1, x2, tmpx;
+    int      i;
+    
+    for(count/=SAMPLING_FACTOR; count; count--)
+    {   tmpx = 0;
+
+        for(i=SAMPLING_FACTOR/2; i; i--)
+        {  x1 = *srce++ / bitshift;
+           x2 = *srce++ / bitshift;
+
+           x1 = (x1 > 32767) ? 32767 : (x1 < -32768) ? -32768 : x1;
+           x2 = (x2 > 32767) ? 32767 : (x2 < -32768) ? -32768 : x2;
+
+           tmpx += x1 + x2;
+        }
+
+        *dste++ = tmpx / SAMPLING_FACTOR;
+    }
+}
+
+
+static void Mix32To16_Stereo(SWORD *dste, SLONG *srce, SLONG count)
+{
+    SLONG   x1, x2, x3, x4, tmpx, tmpy;
+    int     i;
+
+    for(count/=SAMPLING_FACTOR; count; count--)
+    {   tmpx = tmpy = 0;
+
+        for(i=SAMPLING_FACTOR/2; i; i--)
+        {   x1 = *srce++ / bitshift;
+            x2 = *srce++ / bitshift;
+            x3 = *srce++ / bitshift;
+            x4 = *srce++ / bitshift;
+
+            x1 = (x1 > 32767) ? 32767 : (x1 < -32768) ? -32768 : x1;
+            x2 = (x2 > 32767) ? 32767 : (x2 < -32768) ? -32768 : x2;
+            x3 = (x3 > 32767) ? 32767 : (x3 < -32768) ? -32768 : x3;
+            x4 = (x4 > 32767) ? 32767 : (x4 < -32768) ? -32768 : x4;
+
+            tmpx += x1 + x3;
+            tmpy += x2 + x4;
+        }
+
+        *dste++ = tmpx / SAMPLING_FACTOR;
+        *dste++ = tmpy / SAMPLING_FACTOR;
+    }
+}
+
+
+static void Mix32To8_Normal(SBYTE *dste, SLONG *srce, SLONG count)
+{
+    int    x1, x2;
+    int    i, tmpx;
+    
+    for(count/=SAMPLING_FACTOR; count; count--)
+    {   tmpx = 0;
+
+        for(i=SAMPLING_FACTOR/2; i; i--)
+        {   x1 = *srce++ / bitshift;
+            x2 = *srce++ / bitshift;
+
+            x1 = (x1 > 127) ? 127 : (x1 < -128) ? -128 : x1;
+            x2 = (x2 > 127) ? 127 : (x2 < -128) ? -128 : x2;
+
+            tmpx += x1 + x2;
+        }
+
+        *dste++ = (tmpx / SAMPLING_FACTOR) + 128;
+    }
+}
+
+
+static void Mix32To8_Stereo(SBYTE *dste, SLONG *srce, SLONG count)
+{
+    int    x1, x2, x3, x4;
+    int    i, tmpx, tmpy;
+
+    for(count/=SAMPLING_FACTOR; count; count--)
+    {   tmpx = tmpy = 0;
+
+        for(i=SAMPLING_FACTOR/2; i; i--)
+        {   x1 = *srce++ / bitshift;
+            x2 = *srce++ / bitshift;
+            x3 = *srce++ / bitshift;
+            x4 = *srce++ / bitshift;
+    
+            x1 = (x1 > 127) ? 127 : (x1 < -128) ? -128 : x1;
+            x2 = (x2 > 127) ? 127 : (x2 < -128) ? -128 : x2;
+            x3 = (x3 > 127) ? 127 : (x3 < -128) ? -128 : x3;
+            x4 = (x4 > 127) ? 127 : (x4 < -128) ? -128 : x4;
+    
+            tmpx += x1 + x3;
+            tmpy += x2 + x4;
+        }
+
+        *dste++ = (tmpx / SAMPLING_FACTOR) + 128;        
+        *dste++ = (tmpy / SAMPLING_FACTOR) + 128;        
+    }
+}
+
+
+static ULONG samples2bytes(ULONG samples)
+{
+    if(vc_mode & DMODE_16BITS) samples <<= 1;
+    if(vc_mode & DMODE_STEREO) samples <<= 1;
+
+    return samples;
+}
+
+
+static ULONG bytes2samples(ULONG bytes)
+{
+    if(vc_mode & DMODE_16BITS) bytes >>= 1;
+    if(vc_mode & DMODE_STEREO) bytes >>= 1;
+
+    return bytes;
+}
+
+
+static void AddChannel(SLONG *ptr, SLONG todo)
+{
+    SDOUBLE  end;
+    SLONG  done;
+    SWORD  *s;
+
+    while(todo > 0)
+    {   // update the 'current' index so the sample loops, or
+        // stops playing if it reached the end of the sample
+
+        if(vnf->flags & SF_REVERSE)
+        {
+            // The sample is playing in reverse
+
+            if((vnf->flags & SF_LOOP) && (vnf->current < idxlpos))
+            {
+                // the sample is looping, and it has
+                // reached the loopstart index
+
+                if(vnf->flags & SF_BIDI)
+                {
+                    // sample is doing bidirectional loops, so 'bounce'
+                    // the current index against the idxlpos
+
+                    vnf->current    = idxlpos + (idxlpos - vnf->current);
+                    vnf->flags     &= ~SF_REVERSE;
+                    vnf->increment  = -vnf->increment;
+                } else
+                    // normal backwards looping, so set the
+                    // current position to loopend index
+
+                   vnf->current = idxlend - (idxlpos-vnf->current);
+            } else
+            {
+                // the sample is not looping, so check
+                // if it reached index 0
+
+                if(vnf->current < 0)
+                {
+                    // playing index reached 0, so stop
+                    // playing this sample
+
+                    vnf->current = 0;
+                    vnf->active  = 0;
+                    break;
+                }
+            }
+        } else
+        {
+            // The sample is playing forward
+
+            if((vnf->flags & SF_LOOP) && (vnf->current > idxlend))
+            {
+                 // the sample is looping, so check if
+                 // it reached the loopend index
+
+                 if(vnf->flags & SF_BIDI)
+                 {
+                     // sample is doing bidirectional loops, so 'bounce'
+                     //  the current index against the idxlend
+
+                     vnf->flags    |= SF_REVERSE;
+                     vnf->increment = -vnf->increment;
+                     vnf->current   = idxlend-(vnf->current-idxlend);
+                 } else
+                     // normal backwards looping, so set the
+                     // current position to loopend index
+
+                     vnf->current = idxlpos + (vnf->current-idxlend);
+            } else
+            {
+                // sample is not looping, so check
+                // if it reached the last position
+
+                if(vnf->current > idxsize)
+                {
+                    // yes, so stop playing this sample
+
+                    vnf->current = 0;
+                    vnf->active  = 0;
+                    break;
+                }
+            }
+        }
+
+        if(!(s=Samples[vnf->handle]))
+        {   vnf->current = 0;
+            vnf->active  = 0;
+            break;
+        }
+
+        end = (vnf->flags & SF_REVERSE) ? 
+                (vnf->flags & SF_LOOP) ? idxlpos : 0 :
+                (vnf->flags & SF_LOOP) ? idxlend : idxsize;
+
+        done = MIN((end - vnf->current) / vnf->increment + 1, todo);
+
+        if(!done)
+        {   vnf->active = 0;
+            break;
+        }
+
+        if(vnf->vol)
+        {   if(vc_mode & DMODE_STEREO)
+                if(vnf->pan == PAN_SURROUND && vc_mode&DMODE_SURROUND)
+                    vnf->current = MixStereoSurround(s,ptr,vnf->current,vnf->increment,done);
+                else
+                    vnf->current = MixStereoNormal(s,ptr,vnf->current,vnf->increment,done);
+            else
+                vnf->current = MixMonoNormal(s,ptr,vnf->current,vnf->increment,done);
+        }
+        todo -= done;
+        ptr  += (vc_mode & DMODE_STEREO) ? (done<<1) : done;
+    }
+}
+
+
+void VC2_WriteSamples(SBYTE *buf, ULONG todo)
+{
+    int    left, portion = 0;
+    SBYTE  *buffer;
+    int    t;
+    int    pan, vol;
+
+    todo *= SAMPLING_FACTOR;
+
+    while(todo)
+    {   if(TICKLEFT==0)
+        {   if(vc_mode & DMODE_SOFT_MUSIC) md_player();
+            TICKLEFT = (md_mixfreq*125l*SAMPLING_FACTOR) / (md_bpm*50l);
+            TICKLEFT &= ~(SAMPLING_FACTOR-1);
+        }
+
+        left = MIN(TICKLEFT, todo);
+        
+        buffer    = buf;
+        TICKLEFT -= left;
+        todo     -= left;
+
+        buf += samples2bytes(left) / SAMPLING_FACTOR;
+
+        while(left)
+        {   portion = MIN(left, samplesthatfit);
+            memset(VC2_TICKBUF, 0, portion << ((vc_mode & DMODE_STEREO) ? 3 : 2));
+
+            for(t=0; t<vc_softchn; t++)
+            {   vnf = &vinf[t];
+
+                if(vnf->kick)
+                {   vnf->current = (SDOUBLE)(vnf->start) << FRACBITS;
+                    vnf->kick    = 0;
+                    vnf->active  = 1;
+                }
+                
+                if(vnf->frq == 0) vnf->active = 0;
+                
+                if(vnf->active)
+                {   vnf->increment = ((SDOUBLE)(vnf->frq) << (FRACBITS-SAMPLING_SHIFT)) / (SDOUBLE)md_mixfreq;
+                    if(vnf->flags & SF_REVERSE) vnf->increment=-vnf->increment;
+
+                    vol = vnf->vol;  pan = vnf->pan;
+
+                    if((vc_mode & DMODE_STEREO) && (pan!=PAN_SURROUND))
+                    {   lvolsel = (vol * (255-pan)) >> 8;
+                        rvolsel = (vol * pan) >> 8;
+                    } else
+                        lvolsel = (vol*256l) / 480;
+
+                    idxsize = (vnf->size)   ? ((SDOUBLE)(vnf->size) << FRACBITS)-1 : 0;
+                    idxlend = (vnf->repend) ? ((SDOUBLE)(vnf->repend) << FRACBITS)-1 : 0;
+                    idxlpos = (SDOUBLE)(vnf->reppos) << FRACBITS;
+                    AddChannel(VC2_TICKBUF, portion);
+                }
+            }
+
+            if(md_reverb) MixReverb(VC2_TICKBUF, portion);
+
+            if(vc_mode & DMODE_16BITS)
+                Mix32to16((SWORD *) buffer, VC2_TICKBUF, portion);
+            else
+                Mix32to8((SBYTE *) buffer, VC2_TICKBUF, portion);
+
+            buffer += samples2bytes(portion) / SAMPLING_FACTOR;
+            left   -= portion;
+        }
+    }
+}
+
+
+void VC2_SilenceBytes(SBYTE *buf, ULONG todo)
+
+//  Fill the buffer with 'todo' bytes of silence (it depends on the mixing
+//  mode how the buffer is filled)
+
+{
+    // clear the buffer to zero (16 bits
+    // signed ) or 0x80 (8 bits unsigned)
+
+    if(vc_mode & DMODE_16BITS)
+        memset(buf,0,todo);
+    else
+        memset(buf,0x80,todo);
+}
+
+
+ULONG VC2_WriteBytes(SBYTE *buf, ULONG todo)
+
+//  Writes 'todo' mixed SBYTES (!!) to 'buf'. It returns the number of
+//  SBYTES actually written to 'buf' (which is rounded to number of samples
+//  that fit into 'todo' bytes).
+
+{
+    if(vc_softchn == 0)
+    {   VC2_SilenceBytes(buf,todo);
+        return todo;
+    }
+
+    todo = bytes2samples(todo);
+    VC2_WriteSamples(buf,todo);
+
+    return samples2bytes(todo);
+}
+
+
+BOOL VC2_PlayStart(void)
+{
+    if(vc_softchn > 0)
+    {   bitshift = vc_softchn + 257;
+        if(!(vc_mode & DMODE_16BITS))
+            bitshift *= 256;
+        if(md_reverb) bitshift++;
+    }
+
+    samplesthatfit = TICKLSIZE;
+    if(vc_mode & DMODE_STEREO) samplesthatfit >>= 1;
+    TICKLEFT = 0;
+
+    /*  Original Reverb Code!
+        The stuff I use avoids floating point (below).
+
+    RVc1 = (SLONG)(500      * md_mixfreq) / 11000;
+    RVc2 = (SLONG)(507.8125 * md_mixfreq) / 11000;
+    RVc3 = (SLONG)(531.25   * md_mixfreq) / 11000;
+    RVc4 = (SLONG)(570.3125 * md_mixfreq) / 11000;
+    RVc5 = (SLONG)(625      * md_mixfreq) / 11000;
+    RVc6 = (SLONG)(695.3125 * md_mixfreq) / 11000;
+    RVc7 = (SLONG)(781.25   * md_mixfreq) / 11000;
+    RVc8 = (SLONG)(882.8125 * md_mixfreq) / 11000;
+    ReverbPct = 99 - (md_reverb*2);
+    */
+
+    RVc1 = (5000L * md_mixfreq) / REVERBERATION;
+    RVc2 = (5078L * md_mixfreq) / REVERBERATION;
+    RVc3 = (5313L * md_mixfreq) / REVERBERATION;
+    RVc4 = (5703L * md_mixfreq) / REVERBERATION;
+    RVc5 = (6250L * md_mixfreq) / REVERBERATION;
+    RVc6 = (6953L * md_mixfreq) / REVERBERATION;
+    RVc7 = (7813L * md_mixfreq) / REVERBERATION;
+    RVc8 = (8828L * md_mixfreq) / REVERBERATION;
+   
+    if((RVbuf1 = (SLONG *)_mm_calloc((RVc1+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf2 = (SLONG *)_mm_calloc((RVc2+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf3 = (SLONG *)_mm_calloc((RVc3+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf4 = (SLONG *)_mm_calloc((RVc4+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf5 = (SLONG *)_mm_calloc((RVc5+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf6 = (SLONG *)_mm_calloc((RVc6+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf7 = (SLONG *)_mm_calloc((RVc7+1),sizeof(SLONG))) == NULL) return 1;
+    if((RVbuf8 = (SLONG *)_mm_calloc((RVc8+1),sizeof(SLONG))) == NULL) return 1;
+    
+    if(vc_mode & DMODE_STEREO)
+    {   if((RVbuf9 = (SLONG *)_mm_calloc((RVc1+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf10 = (SLONG *)_mm_calloc((RVc2+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf11 = (SLONG *)_mm_calloc((RVc3+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf12 = (SLONG *)_mm_calloc((RVc4+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf13 = (SLONG *)_mm_calloc((RVc5+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf14 = (SLONG *)_mm_calloc((RVc6+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf15 = (SLONG *)_mm_calloc((RVc7+1),sizeof(SLONG))) == NULL) return 1;
+        if((RVbuf16 = (SLONG *)_mm_calloc((RVc8+1),sizeof(SLONG))) == NULL) return 1;
+    }
+    
+    RVRindex = 0;
+    return 0;
+}
+
+
+void VC2_PlayStop(void)
+{
+    if(RVbuf1  != NULL) free(RVbuf1);
+    if(RVbuf2  != NULL) free(RVbuf2);
+    if(RVbuf3  != NULL) free(RVbuf3);
+    if(RVbuf4  != NULL) free(RVbuf4);
+    if(RVbuf5  != NULL) free(RVbuf5);
+    if(RVbuf6  != NULL) free(RVbuf6);
+    if(RVbuf7  != NULL) free(RVbuf7);
+    if(RVbuf8  != NULL) free(RVbuf8);
+    if(RVbuf9  != NULL) free(RVbuf9);
+    if(RVbuf10 != NULL) free(RVbuf10);
+    if(RVbuf11 != NULL) free(RVbuf11);
+    if(RVbuf12 != NULL) free(RVbuf12);
+    if(RVbuf13 != NULL) free(RVbuf13);
+    if(RVbuf14 != NULL) free(RVbuf14);
+    if(RVbuf15 != NULL) free(RVbuf15);
+    if(RVbuf16 != NULL) free(RVbuf16);
+
+    RVbuf1  = NULL;  RVbuf2  = NULL;  RVbuf3  = NULL;  RVbuf4  = NULL;
+    RVbuf5  = NULL;  RVbuf6  = NULL;  RVbuf7  = NULL;  RVbuf8  = NULL;
+    RVbuf9  = NULL;  RVbuf10 = NULL;  RVbuf11 = NULL;  RVbuf12 = NULL;
+    RVbuf13 = NULL;  RVbuf14 = NULL;  RVbuf15 = NULL;  RVbuf16 = NULL;
+}
+
+
+BOOL VC2_Init(void)
+{
+    _mm_errno = MMERR_INITIALIZING_MIXER;
+    if((Samples = (SWORD **)calloc(MAXSAMPLEHANDLES, sizeof(SWORD *))) == NULL) return 1;
+    if(VC2_TICKBUF==NULL) if((VC2_TICKBUF=(SLONG *)malloc((TICKLSIZE+32) * sizeof(SLONG))) == NULL) return 1;
+
+    if(md_mode & DMODE_STEREO)
+    {   Mix32to16  = Mix32To16_Stereo;
+        Mix32to8   = Mix32To8_Stereo;
+        MixReverb  = MixReverb_Stereo;
+    } else
+    {   Mix32to16  = Mix32To16_Normal;
+        Mix32to8   = Mix32To8_Normal;
+        MixReverb  = MixReverb_Normal;
+    }
+
+    vc_mode = md_mode;
+
+    _mm_errno = 0;
+    return 0;
+}
+
+
+void VC2_Exit(void)
+
+// Yay, he joys and fruits of C and C++ -
+//                        Deallocation of arrays!
+
+{
+    //if(VC2_TICKBUF!=NULL) free(VC2_TICKBUF);
+    if(vinf!=NULL) free(vinf);
+    if(Samples!=NULL) free(Samples);
+
+//    VC2_TICKBUF     = NULL;
+    vinf            = NULL;
+    Samples         = NULL;
+}
+
+
+BOOL VC2_SetNumVoices(void)
+{
+    int t;
+
+    if((vc_softchn = md_softchn) == 0) return 0;
+
+    if(vinf!=NULL) free(vinf);
+    if((vinf = _mm_calloc(sizeof(VINFO),vc_softchn)) == NULL) return 1;
+    
+    for(t=0; t<vc_softchn; t++)
+    {   vinf[t].frq = 10000;
+        vinf[t].pan = (t & 1) ? 0 : 255;
+    }
+
+    return 0;
+}
+
+
+void VC2_VoiceSetVolume(UBYTE voice, UWORD vol)
+{
+    vinf[voice].vol = vol;
+}
+
+
+void VC2_VoiceSetFrequency(UBYTE voice, ULONG frq)
+{
+    vinf[voice].frq = frq;
+}
+
+
+void VC2_VoiceSetPanning(UBYTE voice, ULONG pan)
+{
+    vinf[voice].pan = pan;
+}
+
+
+void VC2_VoicePlay(UBYTE voice, SWORD handle, ULONG start, ULONG size, ULONG reppos, ULONG repend, UWORD flags)
+{
+    vinf[voice].flags    = flags;
+    vinf[voice].handle   = handle;
+    vinf[voice].start    = start;
+    vinf[voice].size     = size;
+    vinf[voice].reppos   = reppos;
+    vinf[voice].repend   = repend;
+    vinf[voice].kick     = 1;
+}
+
+
+void VC2_VoiceStop(UBYTE voice)
+{
+    vinf[voice].active = 0;
+}  
+
+
+BOOL VC2_VoiceStopped(UBYTE voice)
+{
+    return(vinf[voice].active==0);
+}
+
+
+void VC2_VoiceReleaseSustain(UBYTE voice)
+{
+
+}
+
+
+SLONG VC2_VoiceGetPosition(UBYTE voice)
+{
+    return(vinf[voice].current>>FRACBITS);
+}
+
+
+/**************************************************
+***************************************************
+***************************************************
+**************************************************/
+
+
+void VC2_SampleUnload(SWORD handle)
+{
+    void *sampleadr = Samples[handle];
+
+    free(sampleadr);
+    Samples[handle] = NULL;
+}
+
+
+SWORD VC2_SampleLoad(SAMPLOAD *sload, int type, FILE *fp)
+{
+    SAMPLE *s = sload->sample;
+    int    handle;
+    ULONG  t, length,loopstart,loopend;
+
+    if(type==MD_HARDWARE) return -1;
+
+    // Find empty slot to put sample address in
+    for(handle=0; handle<MAXSAMPLEHANDLES; handle++)
+        if(Samples[handle]==NULL) break;
+
+    if(handle==MAXSAMPLEHANDLES)
+    {   _mm_errno = MMERR_OUT_OF_HANDLES;
+        return -1;
+    }
+
+    length    = s->length;
+    loopstart = s->loopstart;
+    loopend   = s->loopend;
+
+    SL_SampleSigned(sload);
+
+    SL_Sample8to16(sload);
+    if((Samples[handle]=(SWORD *)malloc((length+16)<<1))==NULL)
+    {   _mm_errno = MMERR_SAMPLE_TOO_BIG;
+        return -1;
+    }
+    // read sample into buffer.
+    SL_Load(Samples[handle],sload,length);
+
+    // Unclick samples:
+
+    if(s->flags & SF_LOOP)
+    {   if(s->flags & SF_BIDI)
+            for(t=0; t<16; t++) Samples[handle][loopend+t] = Samples[handle][(loopend-t)-1];
+        else
+            for(t=0; t<16; t++) Samples[handle][loopend+t] = Samples[handle][t+loopstart];
+    } else
+        for(t=0; t<16; t++) Samples[handle][t+length] = 0;
+
+    return handle;
+}
+
+
+ULONG VC2_SampleSpace(int type)
+{
+    return vc_memory;
+}
+
+
+ULONG VC2_SampleLength(int type, SAMPLE *s)
+{
+    return (s->length * ((s->flags & SF_16BITS) ? 2 : 1)) + 16;
+}
+
+
+/**************************************************
+***************************************************
+***************************************************
+**************************************************/
+